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超 值 、 大 容量 DVD 光 盘 

本 书 实例 源 文件 及 10 小 时 高 清 配套 教学 视频 

@ 12 个 PHP 典 型 模块 源 程序 及 8 小 时 教学 视频 ( 赠送 ) 

@ 6 个 PHP 大 型 项 目 案例 源 程序 及 5 小 时 教学 视频 ( 赠送 ) 
@ 25.5 小 时 MySQL 入 门 教学 视频 ( 赠送 ) 
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内 容 简 介 


本 书 以 简单 、 轻 松 的 语言 细致 地 介绍 了 PHP 开发 的 相关 知识 。 书 中 的 每 章 内 容 都 是 PHP 开发 的 重点 。 
本 书 讲解 由 浅 入 深 ， 通 过 大 量 实例 和 详细 的 代码 及 代码 注释 让 读者 理解 和 掌握 相应 的 知识 点 ， 并 提供 了 
大 量 习 题 供 读者 演练 ， 以 检测 和 巩固 学 习 效 果 。 另 外 ， 作 者 专门 为 本 书 录制 了 配套 多 媒体 教学 视频 ， 以 
辅助 读者 高 效 、 直 观 地 学 习 。 这 些 视频 及 本 书 涉及 的 源 代码 一 起 收录 于 配 书 光盘 中 。 

本 书 共 14 章 , 分 为 5 篇 。 第 1 篇 为 初 识 PHP 脚本 语言 ,介绍 了 PHP 的 基本 定义 、 特点、 原理 及 PHP 
文件 等 ; 第 2 篇 介绍 了 PHP 中 的 常量 、 变 量 和 数据 等 相关 知识 点 ; 第 3 篇 为 PHP 编程 基础 ， 介 绍 了 条 件 
与 循环 语句 、 脚 本 的 重用 、Web 编程 基础 及 数据 的 存储 等 ;第 4 篇 为 面向 对 象 编 程 ， 介 绍 了 PHP 与 操作 
系统 、PHP 与 基于 对 象 的 编程 (OOP) 、PHP 与 MVC 等 ; 第 5 篇 为 开源 PHP 应 用 , 主要 介绍 了 WordPress 
和 Drupal 两 个 常见 开源 PHP 应 用 。 

本 书 着 重 夯实 基础 ， 基 本 覆盖 了 PHP 开发 的 基础 知识 ,特别 适合 PHP 初学 者 打 好 基本 功 这 个 阶段 时 
阅读 ， 也 适合 有 一 定 开 发 经 验 的 读者 查阅 和 参考 。 另 外 ， 本 书 还 适合 大 中 专 院 校 作 为 相关 专业 的 教材 。 
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PHP (Hypertext Preprocessor， 超 文本 预 处 理 语 言 ) 承担 的 任务 是 在 用 户 向 Web 服务 
器 发 出 超 文本 请 求 之 后 ， 在 Web 服务 器 向 用 户 返 回 被 请 求 的 页 面 之 前 ， 由 PHP 引擎 来 对 
用 户 请 求 的 页 面 进行 预 处 理 。 它 是 一 种 HTML 内 和 崔 式 的 语言 ， 可 以 很 方便 地 被 嵌入 到 
HTML 页 面 中 。 这 一 点 与 ASP 类似。 但 是 ，PHP 的 开源 特性 、 跨 平台 特性 、 小 巧 的 体积 和 
高 效 的 运行 效率 ， 以 及 开源 社区 的 广泛 支持 和 它 支持 众多 的 数据 库 等 特性 却 是 它 被 广泛 使 
用 的 根本 原因 。 


1. 开源 、 免 费 


PHP 发 布 于 1995 年 。 从 那 之 后 ， 发 展 非常 迅速 。 这 都 归功 于 它 的 开源 特性 。 在 开源 
协议 的 框架 下 ， 用户 甚 至 不 需 任何 花费 ， 就 可 以 获得 各 种 升级 和 安全 补丁 ,保证 PHP 引擎 
的 安全 。 


2. 广泛 支持 


PHP 的 快速 发 展 也 离 不 开 大 家 的 支持 。 在 互联 网 上 ， 人 们 讨论 最 多 的 编程 语言 就 是 
PHP。 如 果 你 在 使 用 过 程 中 发 现 了 任何 解决 不 了 的 问题 ， 打 开 搜 索引 擎 ， 输 入 问题 的 关键 
词 ， 一 定 可 以 找到 解决 方案 的 。 


3， 支持 众多 的 数据 库 


PHP 对 数据 库 的 支持 也 是 其 深 受 欢迎 的 原因 之 一 。PHP 支持 通过 ODBC 连接 各 种 类 型 
的 数据 库 ， 同 时 也 针对 MySQL 数据 库 开 发 了 便捷 的 连接 和 操作 方式 。 由 于 其 对 MySQL 
数据 库 的 原生 支持 ， 使 MySQL 数据 库 成 为 了 最 受 欢迎 的 开源 数据 库 软件 。 


本 书 特色 


1. 该 谐 幽默 ， 接 地 气 儿 


本 书 试图 以 生动 活泼 的 语言 、 接 地 气 儿 的 讲解 方式 来 为 读者 呈现 PHP 的 世界 。 本 书 讲 
解 知 识 点 时 和 日 常生 活 中 的 方方面面 联系 起 来 ， 让 读者 更 加 容易 理解 抽象 的 编程 世界 。 作 
者 希望 通过 这 种 接地 气 儿 的 语言 和 内 容 组 织 形式 ， 让 读者 在 程序 员 的 道路 上 不 至 于 越 走 越 
累 ， 从 而 充分 领略 到 PHP 的 魅力 。 

2. 夯实 基础 ， 注 重 实用 


本 书 涵盖 读者 初 涉 PHP 开发 所 需要 掌握 的 各 种 基础 知识 。 通 过 本 书 ， 可 以 为 读者 后 续 
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的 PHP 应 用 开发 夯实 基础 。 另 外 ， 本 书 讲解 知识 点 时 列举 了 大 量 生动 有 趣 的 实例 ， 可 以 大 
大 提高 读者 的 实际 编程 能 力 。 

3. 代码 精 讲 ， 注 释 丰 富 

本 书 中 的 代码 大 部 分 给 出 了 详细 的 解释 ， 并且 进行 了 丰富 的 注释 ， 便 于 读者 阅读 和 理 
解 ， 也 可 以 培养 读者 从 一 开始 就 养 成 良好 的 编码 习惯 。 

4. 实例 真实 ， 生 动 有 趣 

就 像 作 家 写 小 说 需要 从 实际 生活 中 汲取 生活 经 验 一 样 ， 程 序 员 在 编写 程序 的 过 程 中 也 


需要 从 实际 生活 经 验 中 找 灵感 。 本 书 中 的 实例 大 多 来 源 于 日 常生 活 。 这 些 例 子 生动 有 趣 ， 
可 以 提升 读者 对 日 常生 活 的 观察 力 ， 从 而 在 日 常生 活 中 找到 编程 的 灵感 。 


5. 视频 教学 ， 高 效 直观 


作者 专门 为 本 书 录制 了 配套 多 媒体 教学 视频 ,以 帮助 读者 更 加 直观 、 高 效 地 阅读 本 书 ， 
达到 事半功倍 的 学 习 效 果 。 


6 实战 练习 ， 巩 固 提高 

本 书 除了 第 14 章 之 外 ， 其 他 各 章 最 后 都 提供 了 练习 题 ， 可 以 帮助 读者 巩固 和 提高 所 
学 的 知识 ， 也 可 以 方便 相关 老师 教学 时 使 用 。 
本 书 内 容 及 体系 结构 


第 1 篇 初 识 PHP 脚本 语言 (第 1~3 章 ) 


本 篇 介绍 了 PHP 脚本 的 工作 原理 ， 并 通过 在 Windows 平台 上 搭建 PHP 开发 环境 进 一 
步 加 深 读者 对 PHP 脚本 语言 工作 原理 的 认识 。 


第 2 篇 常量、 变量 与 数组 (第 4~6 章 ) 
本 篇 介绍 了 PHP 脚本 中 经 常 使 用 的 常量 和 变量 ,以 及 作为 变量 集合 的 数组 。 在 这 三 章 


中 ， 读 者 可 以 了 解 这 些 概念 背后 的 意义 及 何 时 需要 使 用 它们 。 本 篇 通过 大 量 的 示例 和 有 的 
放 矢 的 实战 练习 ， 帮 助 读者 更 好 地 掌握 这 些 知识 。 


第 3 篇 PHP 编程 基础 (第 7~10 章 ) 


本 篇 介绍 了 PHP 中 的 执行 流程 和 控制 机 制 。 通过 这 些 控制 机 制 ,可 以 实现 在 指定 条 件 
下 运行 相应 的 脚本 、 循 环 使 用 脚本 及 脚本 的 大 规模 复 用 。 有 了 这 些 流程 控制 机 制 ， 程 序 员 
便 可 以 用 较 少 的 代码 来 完成 各 种 纷繁 复杂 的 任务 ， 提 高 开发 效率 。 另 外 ， 本 篇 还 介绍 了 
Web 编程 的 基础 和 数据 存储 的 相关 知识 ， 为 使 用 PHP 进行 Web 开发 打 好 基础 。 


第 4 篇 面向 对 象 编程 〈 第 11~13 章 ) 
本 篇 主要 介绍 了 PHP 与 操作 系统 、PHP 与 基于 对 象 的 编程 (OOP) 及 PHP 与 MVC 


前 言 


框架 等 内 容 。 通 过 这 三 章 内 容 的 学 习 , 读者 可 以 更 加 深刻 地 认识 PHP 面向 对 象 编程 的 特性 。 
第 5 篇 开源 PHP 应 用 (第 14 章 ) 


本 篇 主要 介绍 了 WordPress 和 Drupal 两 款 知名 的 PHP 应 用 。 读 者 在 掌握 了 前 面 各 章 
内 容 后 ， 可 以 使 用 这 两 款 PHP 应 用 来 快速 搭建 自己 的 网 站 。 


本 书 超 值 DVD 光盘 内 容 


口 本 书 各 章 涉 及 的 实例 源 文件 ; 

口 10 小 时 本 书 配套 教学 视频 ; 

口 12 个 PHP 典型 模块 源 程序 及 8 小 时 教学 视频 ; 

口 6 个 PHP 大 型 项 目 案例 源 程序 及 5 小 时 教学 视频 ; 
口 25.5 小 时 MySQL 入 门 教学 视频 。 


本 书 读者 对 象 


洲 


口 没有 任何 基础 的 PHP 初学 者 ; 
口 PHP 开发 的 爱好 者 ; 

口 刚 入 职 的 PHP 初中 级 程序 员 ; 
口 大 中 专 院 校 的 师 生 ; 

口 相关 培训 学 校 的 学 员 。 


本 书 作 者 


本 书 由 涂 文 家 主笔 编写 ， 其 他 参与 编写 的 人 员 有 丁 士 锋 、 胡 可 、 姜 永 艳 、 新 鲍 鹏 、 了 和 
、 马 林 、 明 廷 堂 、 牛 艳 霞 、 孙 泽 军 、 王 丽 、 吴 绍兴 、 杨 宇 、 游 梁 、 张 建 林 、 张 起 栋 、 张 
、 郑 伟 、 郑 玉 晖 、 朱 雪琴 、 戴 思 齐 、 丁 航 峰 。 
虽然 笔者 花费 了 大 量 精 力 写作 本 书 ， 并 力图 将 疏 漏 减少 到 最 少 ， 但 仍 钨 百 密 一 琉 。 如 
果 您 在 阅读 本 书 的 过 程 中 发 现 有 任何 疏漏 ， 或 者 对 本 书 讲解 有 任何 疑问 ， 都 可 以 与 作者 取 
得 联系 。E-mail 地 址 : bookservice2008@163.com。 
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第 1 章 什么 是 PHP 


当 你 拿 起 这 本 书 的 时 候 ， 起 码 应 该 听 说 过 PHP。 无 论 你 是 否 明白 这 三 个 字母 分 别 代表 
哪儿 个 单词 ,起 码 应 该 对 PHP 有 一 些 了 解 ， 并 想 深入 学 习 这 门 脚本 语言 。 本 章 将 围绕 PHP 
的 定义 、 为 什么 要 使 用 PHP 及 PHP 是 如 何 工作 的 这 几 个 问题 来 展开 讨论 。 


1.1 PHP 的 定义 


按照 PHP 官方 网 站 首页 (http://www.php.net) 给 出 的 定义 ，PHP 是 一 种 通用 的 脚本 语 
言 。 因 其 特别 适合 Web 开发 ， 并 且 可 以 嵌入 HTML 语言 而 得 到 广泛 的 使 用 。 维 基 百 科 则 
指出 ，PHP 是 最 早出 现 的 几 种 可 以 柑 入 HTML 源 文 件 的 服务 器 端 脚本 语言 之 一 。 安 装 了 
PHP 处 理 模块 的 Web 服务 器 可 以 通过 解析 PHP 代码 来 动态 地 生成 最 终 的 页 面 。 简 而 言 之 ， 
PHP 是 一 种 开发 动态 网 页 的 通用 服务 器 端 脚本 语言 。 

在 这 一 段 定 义 PHP 的 文字 中 ， 有 这 样 几 个 关键 词 : 动态 网 页 、 通 用 编程 语言 和 服务 器 
端 脚本 语言 。 在 本 节 中 ， 将 围绕 这 三 个 概念 及 与 之 相对 应 的 一 些 概念 来 了 解 什么 是 PHP。 


1.1.1 动态 网 页 vs. 静态 网 页 


根据 上 文 的 定义 ，PHP 是 一 种 用 于 开发 动态 网 页 的 脚本 语言 。 那 么 ， 什 么 是 动态 网 页 
呢 ? 如 果 存 在 动态 网 页 ， 那 么 肯定 有 一 种 网 页 叫 静 态 网 页 ， 那 么 ， 什 么 又 是 静态 网 页 呢 ? 
本 小 节 主要 讨论 这 两 个 问题 。 

1. 动态 网 页 


维基 百科 将 动态 网 页 定义 为 : 在 用 户 访问 时 或 与 用 户 交 互 时 实时 生成 或 发 生 改变 的 网 
页 。 这 一 概念 是 Web 2.0 时 代 的 基础 概念 。 正 是 有 了 动态 网 页 这 个 概念 ， 跨 站 信息 共享 才 
成 为 可 能 。 为 了 更 深入 地 了 解 动态 网 页 的 运作 机 制 ， 可 以 参考 图 1-1。 

本 地 计算 机 服务 器 


服务 器 存储 


图 1-1 动态 网 页 工作 原理 
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在 图 1-1 中 ， 通 过 “本 地 计算 机 ”访问 到 的 网 页 如 果 是 动态 网 页 的 话 ， 那 么 一 般 是 在 
“本 地 计算 机 ”向 “服务 器 ”发 送 请 求 后 ， 由 “服务 器 ”从 服务 器 存储 中 获取 实时 数据 并 钳 
入 用 户 请 求 的 页 面 中 ， 然 后 返回 结果 给 “本 地 计算 机 ”。 

2. 静态 网 页 


与 动态 网 页 会 在 用 户 发 起 访问 时 实时 生成 网 页 不 同 ， 静 态 网 页 所 呈现 的 内 容 与 存储 在 
服务 器 上 的 内 容 是 一 模 一 样 的 ， 不 会 根据 访问 者 发 起 访问 的 时 间 、 地 点 或 者 其 他 因素 的 不 
同 而 发 生变 化 。 静态 网 页 一 般 以 HIML 格式 存储 在 服务 器 上 并 通过 HITP 协议 呈现 给 网 页 
浏览 者 。 


1.1.2 通用 编程 语言 vs. 基于 域 的 编程 语言 


通用 编程 语言 是 指 可 以 用 来 为 不 同 的 应 用 域 (Application Domain ) 编写 应 用 程序 的 编 
程 语言 。 若 干 个 应 用 域 中 的 应 用 程序 即使 是 使 用 同一 种 通用 编程 语言 编写 的 ， 这 些 应 用 程 
序 也 不 会 相互 影响 。 究 其 原因 ， 系 统 会 为 每 一 个 应 用 域 分 配 一 个 独立 的 虚拟 地 址 空间 
(Virtual Address Space) 。 而 操作 系统 就 是 根据 虚拟 地 址 空间 来 为 应 用 程序 分 配 资源 的 。 

与 通用 编程 语言 对 应 的 一 个 概念 就 是 基于 域 的 编程 语言 。 这 一 概念 与 本 书 的 学 习 无 
关 ， 故 略 去 。 读 者 如 果 有 兴趣 继续 探究 ， 请 自行 查阅 相关 资料 。 


1.1.3 ”服务 器 端 脚本 语言 vs. 客户 端 脚本 语言 


根据 定义 可 知 ，PHP 是 一 种 服务 器 端 脚本 语言 。 那 么 在 B/S 架构 下 ， 有 服务 器 端 脚本 
语言 ， 就 一 定 会 有 客户 端 脚本 语言 。 本 小 节 将 对 这 两 种 脚本 语言 的 特点 进行 比较 。 


1. 客户 端 脚本 语言 


客户 端 脚本 语言 是 指 在 一 个 网 页 的 范围 内 根据 鼠标 和 键盘 的 动作 或 某 一 时 间 事 件 动 
态 改 变 网 页 内 容 的 脚本 语言 。 比 较 常 见 的 客户 端 脚本 语言 有 JavaScript 和 ActionScript 等 。 


2. 服务 器 端 脚本 语言 


服务 器 端 脚本 语言 是 指 服务 器 根据 用 户 请 求 做 出 回应 来 改变 网 页 的 若干 内 容 、 调 整 网 
页 载 入 顺序 或 重 载 页 面 。 比 较 常见 的 服务 器 端 脚本 语言 除了 PHP， 还 有 ASP、ASP.NET 
和 JSP。 

通过 对 动态 网 页 、 通 用 编程 语言 和 服务 器 端 脚本 语言 这 三 个 概念 的 理解 ， 我 们 大 致 上 
了 解 了 PHP 脚本 语言 的 功能 与 作用 ， 用 一 句 话 概括 就 是 : 通过 部 署 PHP 处 理 模块 ，Web 
服务 器 在 回应 用 户 请 求 时 可 以 根据 用 户 的 需要 动态 更 新 呈现 给 用 户 的 网 页 。 


1.2 为 什么 要 使 用 PHP 


在 了 解 了 PHP 脚本 语言 的 定义 和 功能 之 后 , PHP 脚本 语言 是 否 是 唯一 的 选择 , 这 一 问 


。3. 
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题 引起 了 我 们 的 兴趣 。 既然 PHP 脚本 语言 并 不 是 唯一 的 一 种 开发 动态 网 页 的 通用 的 服务 器 
端 脚本 语言 ， 那 么 为 什么 要 学 习 并 使 用 PHP 呢 ? 换言之 , PHP 到 底 有 哪些 优点 ， 使 得 人 们 
趋 之 若 仪 呢 ? 


1.2.1 PHP 脚本 的 特点 


1， 响 应 速度 快 


正 是 由 于 PHP 脚本 语言 可 以 内 嵌 于 HTML 代码 之 中 ，Web 服务 器 可 以 在 非常 短 的 时 
间 内 加 载 一 个 PHP 文件 并 解析 文件 中 内 嵌 的 PHP 脚本。 


2. 免费 开源 


PHP 是 一 个 免费 开源 的 项 目 ， 无 论 是 在 Web 服务 器 上 部 团 PHP 引擎 ， 还 是 使 用 PHP 
代码 编写 网 站 ， 用 户 都 不 需要 花费 一 分 钱 。 天 下 到 底 还 是 有 免费 的 午餐 。 


3. 易 用 性 

PHP 脚本 的 语法 简单 明了 ， 即 使 是 对 于 没有 任何 编程 经 验 的 人 来 说 也 算得 上 是 通俗 易 
懂 的 了 。 

4， 跨 平台 支持 

PHP 引擎 支持 种 类 繁多 的 服务 器 操作 系统 ， 如 微软 视窗 操作 系统 、 开 源 的 Linux、 苹 
果 的 Mac OS 及 Unix 操作 系统 的 各 种 变 体 。 


5. 无 处 不 在 的 技术 支持 


PHP 官方 网 站 提供 了 一 份 邮件 讨论 组 列表 。 这 些 讨 论 组 讨论 的 内 容 涵盖 了 诸如 PHP 
概述 、PHP 和 Windows、PHP 和 数据 库 之 类 的 话题 。 读 者 可 以 根据 自己 的 需要 有 选择 地 加 
入 这 些 讨论 组 , 以 便 及 时 的 获取 支持 。 另外 , 互联 网 上 非 官方 的 PHP 资源 也 是 异常 的 丰富 ， 
通过 搜索 引擎 ， 可 以 很 方便 地 找到 学 习 和 使 用 PHP 过 程 中 碰 到 的 各 种 问题 的 答案 。 


6. 无 懈 可 击 的 安全 脚本 


只 要 用 户 编写 的 PHP 脚本 语法 合乎 规范 、 运 行 无 报错 ， 访 问 者 是 不 可 能 宕 见 PHP 代 
码 的 。 这 样 一 来 ， 脚 本 的 安全 性 也 就 得 到 了 保证 。 

7. 强大 的 可 定制 功能 

PHP 的 开源 许可 允许 开发 者 修改 PHP 软件 为 已 所 用 。 

上 面 的 表述 中 罗列 的 特点 虽然 大 致 能 够 反映 PHP 为 什么 受到 追捧 ， 但 是 我 们 对 PHP 
脚本 到 底 能 够 做 些 什么 仍然 一 知 半 解 。 为 了 解决 这 个 问题 ， 下 面 将 分 四 个 小 节 来 讲述 PHP 
脚本 到 底 都 能 做 些 什么 。 


1.2.2 PHP 脚本 和 Web 应 用 程序 


正如 前 文 所 提 到 的 那样 , 在 动态 脚本 出 现 之 前 , 网 站 都 是 由 一 个 个 静态 的 页 面 组 成 的 ， 
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访问 者 通过 访问 页 面 上 的 链接 在 不 同 的 页 面 间 切 换 。 在 那个 时 候 ， 访 问 者 只 能 被 动 地 接受 
网 页 上 的 信息 ， 而 无 法 与 网 站 的 内 容 进 行 交互 ， 访 问 为 他 们 量 身 定做 的 页 面 ， 进 而 实现 信 
息 的 有 效 传播 。 
正 是 基于 这 样 一 种 认识 ， 人 们 开发 出 种 类 各 异 的 脚本 语言 来 实现 网 页 的 动态 化 ，PHP 
脚本 就 是 其 中 之 一 。 由 于 PHP 脚本 内 嵌 在 HTML 代码 之 中 ， 并 在 发 送 给 用 户 之 前 就 已 经 
被 安装 在 Web 服务 器 上 的 PHP 处 理 模 块 处 理 过 了 ， 因 此 用 户 从 Web 服务 器 上 接收 到 的 回 
应 页 面 中 的 PHP 脚本 已 经 被 蔡 换 成 了 相应 的 HTML 代码 ， 从 而 实现 了 动态 响应 用 户 请 求 
的 目的 。 

网 站 开发 者 按照 一 定 的 规划 运用 PHP 脚本 编写 的 网 站 因 具 有 动态 化 、 交 互 性 的 特点 而 
被 称 为 Web 应 用 程序 。 具 体 来 说 ，PHP 脚本 在 Web 应 用 程序 中 主要 从 事 如 下 几 项 工作 

口 与 HTML 表单 进行 交互; 

口 与 数据 库 通 信 ; 

口 生成 安全 的 网 页 。 
虽然 PHP 脚本 有 着 上 述 的 优势 ， 但 是 它 也 不 是 万 能 的 。 因 为 PHP 是 服务 器 端 脚本 语 
言 , 它 不 能 和 用 户 的 浏览 器 进行 交互 , 也 就 无 法 实现 对 鼠标 动作 和 用 户 屏 幕 分 辨 率 的 响应 。 
这 样 一 来 ， 如 果 仅 使 用 PHP 脚本 ， 则 无 法 实现 一 些 诸如 浮动 导航 菜单 之 类 的 特效 。 

因此 在 编写 网 站 时 ,还 要 结合 使 用 客户 端的 脚本 语言 ， 如 JavaScript, 来 提升 网 站 易 用 
性 和 丰富 用 户 体验 。 


1.2.3 ”PHP 脚本 和 数据 库 应 用 


PHP 在 与 数据 库 交 互 方面 有 着 无 与 伦比 的 优势 ， 因 为 它 支持 几乎 所 有 类 型 的 数据 库 ， 
甚至 有 些 是 读者 没有 听 说 过 的 。 你 只 需要 告诉 PHP 引擎 需要 连接 的 数据 库 的 名 字 以 及 数据 
库 的 访问 路 径 ，PHP 引擎 会 帮助 连接 数据 库 ， 将 你 的 指令 传送 到 数据 库 ， 然 后 将 数据 库 的 
处 理 结 果 返 回 给 你 。 

比较 常见 的 数据 库 类 型 有 : dBASE、Informix、Ingres、Microsoft SQL Server、mSQL、 
MySQL、Oracle、PostgreSQL 和 Sybase。 

除了 支持 这 些 常 见 的 数据 库 之 外 ，PHP 还 支持 开放 数据 库 互 联 (ODBC) 。 通 过 开放 
数据 库 互联 ， 你 甚至 可 以 连接 到 PHP 无 法 原生 支持 的 数据 库 ， 比 如 说 Microsoft Access 和 
IBM DB2。 


1.2.4 PHP 脚本 和 文件 系统 


PHP 脚本 还 可 以 和 文件 系统 进行 交互 。 换 名 话说，PHP 可 以 访问 那些 存储 在 本 地 和 其 
他 计算 机 上 可 以 通过 网 络 访问 的 文件 夹 ， 并 在 这 些 文件 夹 里 创建 、 修 改 、 删 除 文件 或 者 改 
变 文 件 属性 。 你 能 够 在 电脑 上 进行 的 各 种 操作 基本 上 都 可 以 通过 PHP 脚本 来 实现 。 

许多 网 络 应 用 程序 都 会 要 求 与 本 地 的 文件 系统 进行 交互 。 举 个 例子 ， 某 网 络 应 用 程序 
在 文件 系统 中 的 Intemet 临时 文件 夹 里 创建 若干 文件 用 来 记录 相关 信息 。 当 用 户 在 同一 次 
会 话 中 再 次 请 求 这 些 信息 时 ， 该 网 络 应 用 程序 会 将 存储 在 本 地 的 信息 发 送 给 用 户 ， 而 不 是 
从 远程 数据 库 中 重新 读 取 数 据 ， 从 而 大 大 地 提高 了 网 站 的 访问 速度 。 
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另外 ， 大 多 数 的 系统 管理 和 维护 脚本 也 会 频繁 地 和 文件 系统 进行 交互 。 比 如 ， 你 可 以 
编写 一 段 PHP 脚本 来 备份 文件 ， 清 理 文 件 夹 或 者 处 理 一 些 文本 文件 。 


1.2.5 PHP 脚本 和 系统 命令 


PHP 脚本 还 可 以 向 操作 系统 发 出 指令 ， 要 求 操作 系统 进行 相关 的 操作 。 例 如 ， 你 可 以 
通过 一 段 PHP 脚本 执行 某 系统 命令 并 获得 该 命令 的 屏 显 信息 。Windows 操作 系统 提供 了 命 
令 行 提示 符 ， 通 过 命令 行 提示 符 执行 dir 命令 来 显示 当前 路 径 下 的 文件 和 文件 夹 的 相关 信 
息 。 同 样 ， 通 过 编写 一 段 PHP 脚本 ， 也 可 以 执行 同样 的 操作 。 

通过 PHP 脚本 执行 操作 系统 命令 通常 会 被 用 来 进行 系统 管理 和 维护 。 例 如 ， 你 想 清理 
某 文件 夹 中 特定 类 型 的 文件 , 那么 你 可 以 使 用 PHP 脚本 发 出 指令 要 求 操作 系统 列 出 该 文件 
夹 中 的 所 有 文件 ， 然 后 识别 并 删除 指定 类 型 的 文件 。 

除 此 之 外 , 通过 PHP 脚本 还 可 以 运行 操作 系统 中 安装 的 各 种 应 用 程序 ,无论 它们 是 否 
是 用 PHP 脚本 编写 的 。 这 样 一 来 ,我们 可 以 有 的 放 矢 的 进行 应 用 程序 开发 : 调用 已 经 存在 
的 功能 ， 开 发 新 的 功能 ， 既 节约 了 资源 ， 又 提高 了 效率 。 


1.3 什么 是 PHP 文件 


在 了 解 了 PHP 脚本 之 后 ， 还 需要 关心 一 个 问题 : 写 好 的 PHP 脚本 应 该 以 什么 形式 保 
存 起 来 呢 ? 保存 好 的 PHP 文件 又 是 如 何 工作 的 呢 ? 阅读 本 节 ， 你 就 可 以 找到 答案 。 


1.3.1 PHP 文件 的 特征 


我 们 可 以 把 编写 好 的 PHP 脚本 保存 在 一 个 扩展 名 为 .php 或 ,phtml 的 文件 中 。 一 个 PHP 
文件 可 以 包含 文本 、HTML 标签 以 及 PHP 脚本 。 在 Web 服务 器 上 安装 了 PHP 处 理 模块 之 
后 ，Web 服务 器 就 可 以 处 理 PHP 脚本 了 。 

在 PHP 文件 中 ， 一 段 PHP 脚本 通常 是 以 “<?php” 开 头 ， 以 “?>” 结 尾 的 。 每 一 行 
PHP 代码 都 需要 以 半角 分 号 结束 ， 如 同 下 面 这 段 代 码 一 样 。 

【 例 1.1】 示例 代码 : 


1 <?php echo "<p>Hello World"; ?> 


在 例 1.1 中 ，“<?php” 是 脚本 的 开始 标签 、“?>” 是 脚本 的 结束 标签 ， 而 “echo” 是 
一 条 PHP 用 来 输出 文本 的 命令 ， 它 告诉 PHP 解析 器 原样 输出 关键 字 后 面 引号 中 的 内 容 。 
Web 服务 器 在 读 取 到 这 段 代码 后 就 会 输出 例 1.2 所 示 并 将 其 发 放 到 用 户 的 浏览 器 中 。 

例 1.2】 示例 代码 : 
1 <p>Hello World 


用 户 在 浏览 器 中 查看 源 代码 时 ， 就 会 看 到 例 1.2 所 示 中 的 代码 ， 而 不 会 看 到 任何 PHP 
语句 。 这 样 一 来 ，PHP 脚本 对 于 用 户 来 说 就 像 空气 一 样 ， 看 不 见 也 摸 不 着 ， 脚 本 的 安全 性 


得 到 了 极 大 的 保障 。 
1.3.2 PHP 文件 是 如 何 工作 的 


那么 是 不 是 任何 一 台 服 务 器 都 可 以 处 理 PHP 文件 呢 ? 答案 当然 是 否定 的 .PHP 和 Web 
服务 器 必须 要 紧密 配合 才能 完成 处 理 PHP 文件 的 任务 。 通 常情 况 下 ，Apache 软件 基金 开 
发 的 Apache HTTP 服务 器 可 以 做 为 Web 服务 器 来 处 理 PHP 文件 ， 而 PHP 与 Apache 服务 
器 的 配合 也 是 十 分 默契 的 。 当 然 ，PHP 服务 也 可 以 安装 在 Microsoft IS 服务 器 上 ， 不 过 配 
置 起 来 要 麻烦 的 多 ， 具 体 的 内 容 会 在 第 2 章 中 讲 到 。 

在 这 一 小 节 里 , 我 们 重点 来 回顾 一 下 上 一 节 中 讲 到 的 PHP 文件 的 处 理 过 程 ,进而 了 解 
PHP 文件 是 如 何 工作 的 。 

PHP 是 一 种 简单 易 读 的 高 级 脚本 语言 ， 人 们 可 以 像 阅 读 人 类 语言 一 样 阅读 PHP 脚本 ， 
然而 电脑 却 没 有 办 法 读 懂 这 些 。 如 果 想 要 电脑 读 懂 PHP 脚本 , 需要 PHP 解析 器 。 通 过 PHP 
解析 器 , 所 有 的 PHP 脚本 都 可 以 被 解析 成 为 电脑 能 读 懂 的 语言 。 只 有 这 样 ， 电 脑 才 能 执行 
通过 PHP 脚本 下 发 的 各 种 指令 。 

当 用 户 通过 浏览 器 向 某 Web 服务 器 发 出 请 求 后 ，Web 服务 器 根据 用 户 的 请 求 打开 请 
求 的 某 PHP 文件 ， 并 在 文件 中 找寻 需要 处 理 的 PHP 脚本 。 当 所 有 的 PHP 脚本 根据 用 户 需 
求 被 转换 成 HTML 代码 后 ，Web 服务 器 将 处 理 结果 , 也 就 是 一 份 包含 用 户 请 求 信息 的 纯粹 
的 HTML 代码 文件 ， 发 送 到 用 户 浏 览 器 。 浏 览 器 再 将 接收 到 的 文件 展示 给 用 户 。 图 1-2 形 
象 地 展示 了 这 一 过 程 。 


PHP 
TE 有 
<html> 
用 户 服务 器 划 
</html> 
HTML 


图 1-2 PHP 文件 处 理 机 制 


1.4 习 题 


(1) 什么 是 动态 网 页 ?什么 是 静态 网 页 ? 它们 有 什么 区 别 ? 
(2) PHP 脚本 有 哪些 特点 ? 
(3) PHP 文件 是 如 何 工 作 的 ? 
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在 上 一 章 中 ， 我 们 了 解 了 PHP 是 一 种 通用 的 服务 器 端 脚本 语言 ， 这 也 就 意味 着 PHP 
的 运行 是 需要 有 服务 器 支持 的 。 通 常情 况 下 ， 网 络 上 会 部 署 至 少 两 台 计算 机 ， 一 台 充 当 服 
务 器 ， 一 台 充 当 客户 端 。 在 充当 服务 器 的 计算 机 上 安装 PHP 处 理 模 块 并 上 传 可 供 客户 端 访 
问 的 网 页 。 当 接收 到 客户 端 发 出 的 访问 请 求 时 ， 服 务 器 便 使 用 PHP 处 理 模块 处 理 被 请 求 页 
面 中 的 PHP 脚本， 然后 将 处 理 完成 的 页 面 返回 至 客户 端的 浏览 器 。 

按照 这 样 的 说 法 , 岂 不 是 要 准备 两 台电 脑 ? 其 实 不 然 ,为 了 完成 PHP 运行 环境 的 搭建 ， 
一 台电 脑 足 矣 。 


2.1 准备 必要 的 文件 


按照 1.3.2 小 节 中 的 描述 ， 至 少 需要 一 个 Web 服务 器 软件 和 PHP 处 理 引擎 。 另 外 ， 为 
了 存储 网 页 交互 产生 的 大 量 数 据 ， 还 需要 搭配 一 个 数据 库 软 件 。 在 本 书 中 ， 所 有 的 PHP 页 
面 都 是 运行 在 Apache 服务 器 、PHP 处 理 引 擎 和 MySQL 数据库 构 建 的 环境 中 的 。 本 节 将 主 
要 讲述 如 何 获取 并 安装 这 些 组 件 。 


2.1.1 获取 Apache HTTP 服务 器 软件 


Apache HTTP 服务 器 是 由 Apache 软件 基金 会 赞助 开发 的 众多 开源 软件 产品 中 的 杰出 
代表 。 读 者 可 以 访问 Apache 软件 基金 会 的 网 站 获取 更 多 关于 基金 会 及 获得 其 赞助 的 软件 
项 目的 有 关 信 息 。 这 里 需要 从 官网 上 下 载 Apache HTTP 服务 器 备用 。 

访问 Apache 软件 基金 会 的 官方 网 站 (http://www.apache.org)， 单 击 页 面 右上 角 的 “项 
目 (Project) ”选项 ,在 打开 的 页 面 中 , 单 击 页 面 左 侧 “ 索 引 (Index) ”项 下 的 “类 别 (Categories)” 
子 项 ， 进 入 Apache 软件 基金 会 项 目 分 类 索引 ， 如 图 2-1 所 示 。 


“Nm Pche Software Foundation 
http:// www.apache.org/ 


2-1 Apache 软件 基金 会 项 目 分 类 索引 
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ho 


外 击 “http ”链接 后 ， 找 到 并 进入 Apache HTTP Server 项 目 主页 ， 如 图 2-2 所 示 。 
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图 2-2 Apache HTTP 服务 器 项 目 主页 


在 主页 中 找到 并 单 击 “ 由 此 下 载 发 行 版 本 (Releases can be downloaded from) ”的 链 
接 进入 软件 下 载 页 面 ， 如 图 2-3 所 示 。 推 荐 下 载 最 新 的 稳定 版 本 2.4.3。 
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图 2-3 ”Apache HTTP 服务 器 下 载 页 面 


2.1.2 获取 PHP 处 理 引擎 


PHP 处 理 引擎 通常 是 两 个 版 本 并 行 开 发 的 ， 截 止 2012 年 10 月 18 日 ，PHP 的 最 新 版 
本 为 5.4.8 和 5.3.18。 官 方 网 站 推荐 将 PHP 处 理 引 擎 升级 到 这 两 个 版 本 的 其 中 之 一 ， 但 是 


。9 。 
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这 两 个 版 本 通常 被 用 来 和 Microsoft IIS 搭档。 因此， 本 书 将 使 用 的 PHP 版 本 为 5.2.17。 
访问 PHP 的 官方 网 站 (http://www.php.net) ， 在 网 站 首页 上 即 可 找到 下 载 入 口 ， 如 图 


2-4 所 示 。 
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图 2-4 PHP 官方 网 站 首页 
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在 首页 的 右 侧 有 三 个 推荐 版 本 的 下 载 链接 ， 同 时 在 导航 栏 上 也 有 一 个 下 载 页 面 的 链 
接 。 单 击 首页 顶端 导航 栏 的 “Downloads” 链 接 可 以 进入 推荐 版 本 的 下 载 页 面 ， 如 果 需 要 
下 载 之 前 的 版 本 ， 可 以 单 击 下 载 页 面 右 侧 的 “Old archives” 链 接 进 入 旧版 本 的 下 载 页 面 。 


本 章 使 用 的 是 5.2.17 版 本 的 PHP 引擎 ， 如 图 2-5 所 示 。 


» Download: PHP Sl (oc be2) PHP Si (tar.c2) 
5,2,16 


» Released: 16 December 2010 
sh 


S216 (tar.bz2) PHP S.2.16 (tar.gz) 
5.2.15 


» Released: 09 December 2010 
。 ‘cement English 


S21s (tarbza) PHP 5.2.15 (car.ez) 


图 2-5 Windows 二 进 制版 PHP 处 理 引 擎 下 载 页 面 


2.1.3 获取 MySQL 数据 库 软 件 


MySQL 数据 库 是 一 款 由 甲 


骨 文 公司 〈Oracle Inc.) 开发 的 快速 、 


多 线程 、 多 用 户 和 健 
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壮 的 数据 库 系 统 。 按 照 MySQL 官方 网 站 (http://dev.mysql.com) 的 说 法 ，MySQL 是 全 球 


最 受 欢迎 的 开源 数据 库 系 统 。 截 止 目前 ，MySQL 的 最 新 版 本 是 5.2.8。 本 书 将 使 月 
本 的 MySQL 数据 库 。 
访问 MySQL 官方 网 站 ， 单 击 首页 上 的 下 载 选 项 卡 ， 进 入 下 载 页 面 。 


i 盏 
AN 总 Login | Register 
MySQL 四 


The world's opular open source database 


“Downioads™ 
Developer Zone aa Documentation 四 日 名 


Current ™ Archives 


和 MysQL Downloads 


MySQL Community Server 


MySQL Cluster 


MySQL Installer 国 New Releases 


MySQL Workbench (GUI Tool) ee MySQL Cluster 7.1 


MySQL Proxy 


AlI MySQL Products. For All Windows Platforms. MYSQL Cluster 7.0 
MySQL Connectors In One Package. 
-一 一 一 一 一 一 一 MysQL Community server 5.1 
MysQL Installer (windows) Download 


Connector/Python 1.0 


MySQL Community Server MysQL Community se 


图 2-6 MySQL 数据 库 下 载 入 口 


单 击 如 图 2-6 所 示 的 下 载 链接 ,进入 MySQL 数据 库 安装 包 的 下 载 页 面 , 如 图 2-7 所 示 。 


单 击 “Download” 按 钮 下 载 安装 包 。 


MySQL D Please report any bugs or inconsistencies you observe to our Bugs Database, 
Newsletter Thank you for your support! 
Over 1 Million Subscriber 


MySQL Installer 5.5.28 


Related Pages: Select platform: 
Technical Articles 


Microsoft Windows 国 seect 


Documentation 


Windows (x86, 32-bit), MSI Installer 5.5.28 207.0M 


ownload 


(mysqHinstaller-community-5.5.28.2.msi) MD5: eff290666a7a714587c425378bc915fc | Signature 


四 we suooest that you use the MDS checksums and GnupG signatures to verify the integrity of the 
packages you download 


MySQL Installer 5.6.7 rc 


Select Platform: 


Microsoft Windows 国 seect 


图 2-7 MySQL 安装 包 下 载 页 面 


2.1.4 ”获取 数据 库 管理 软件 phpMyAdmin 


phpMyAdmin 是 一 款 完全 由 PHP 脚本 编写 而 成 的 、 用 于 在 互联 网 上 管理 MySQL 数据 


日 最 新 版 
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库 的 免费 软件 。 它 直观 的 Web 界面 支持 对 MySQL 数据 的 大 部 分 操作 。 和 截止 目前 ， 
phpMyAdmin 的 最 新 版 本 是 3.5.3， 本 书 将 使 用 phpMyAdmin 的 最 新 版 本 。 

访问 phpMyAdmin 的 官方 网 站 (http:/www.phpmyadmin net) 。 在 页 面 的 右上 方 ， 单 
击 如 图 2-8 所 示 的 红 框 内 的 “.zip” 链 接 转 到 由 SourceForge 提供 的 下 载 页 面 ， 如 图 2-9 
所 示 。 


News Security Support Docs Try Contribute Sponsors Themes Download 


About Download 36:: 
jr zip notes 
phplAdminisaffee sofwaretool wrten in FP., niended to handie the adminisvation of MySCL over he Word Wide 


Web phpMyAdmin suppons a wide fange of operalions with MySQL The most frequently Used operafions are supported 
by the user iteriace [managing databases, tables feids. relafions, indexes, users, pemissions. etch whie you Shi have Ty ph Nn 
the abllty lo directly eyecule any SQL statement 


pnphyAdmin comes win a wide range of nnand users are wecome 10 Update or mk pages to share ideas 村 
and howos for vanous operalions The Php Wity 0 nep you fyou face any Problem YOU can use a | Donate to prouyaamin | 
4 gethep 


re phphyAdmin is also very deeply documertad na book winen by one of the deveiopars — Papa 
de Mastenng orpMyAdmn Ior FTemmwe MYSOL Mamagemart whch Savatlable in Engish Czech a 
AOEm 
To ease usage 10 a wide range ofpeople phpMyAdmin sbeing Yansiated mio nanouagesand 
suppors both LTR and RTL anguages 机 
wm CYBERDAY 


Since version 3.0.0. phpMyAdmin joined the GoPHPS initatve and dropped compatiblity code for 
older PHP and MYSQL versions; version 3 and later requires at least PHP 52 and MySQLS 


图 2-8 ” phpMyAdmin 官方 网 站 首页 


€® CI 
Fevortes | 国 pownoad phpWyAmin trom SourceForge net 乔 - 回 -局 两 、Pege- sofey- Toob- e-| 
BTo help protect your security, Internet Eplorer blocked this site from downlcading files to your computer, Click herefor cptions.. x 
sOurceforge = es | Se 


phpMyAdmin Miror pravided by | 
ERA Yourdomoad mil startin 0 seconds 


frebome wth the downlcad7 Check your browaers secarty bar or ty a rect ok or try arother rer 


图 2-9 ”SourceForge 提供 的 phpMyAdmin 下 载 页 面 


车 读者 使 用 的 浏览 器 是 Windows Internet Explorer, 浏览 器 很 有 可 能 会 出 于 安全 原因 阻 
止 文件 自动 下 载 ， 并 在 页 面 上 方 显示 提示 信息 〈 如 图 2-9 中 所 示 的 黄色 安全 提示 条 ) 。 请 
在 提示 信息 上 单 击 ， 并 在 弹出 的 菜单 中 选择 “下 载 文件 (Download File) ”， 页 面 刷新 后 ， 
弹出 “文件 下 载 (File Download) ”对 话 框 ， 如 图 2-10 所 示 。 


Fie Downioad [一 


Do you want to open or save this fle? 


i Nane: phpMyAdmin3.53-al-languages -ip 
Type: Compressed lipped) Folder, 605MB 
Fem- jaistdl.sourceforge.net 


Ee Ese) Ce 


Nwars ask beiore opening this tye of fle 


各 Whie fies from the temet can be Useful some fies can palertialy 
ham yaar campuiar Fyou do rot tnt ths source do rat open o 
savethefle What'sithe rd? 


图 2-10 文件 下 载 对 话 框 


直接 单 击 “ 保 存 (Save) ”按钮 保存 文件 到 指定 目录 中 备用 。 
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2.2 安装 Apache HTTP 服务 器 


由 于 获取 的 是 Apache HTTP 服务 器 的 Windows 安装 包 ， 因 此 整个 安装 过 程 与 其 他 的 
Windows 应 用 程序 类 似 ， 操 作 起 来 非常 的 简单 。 需 要 注意 的 是 ， 本 书 所 有 的 安装 示例 都 是 
在 Windows 7 英文 旗舰 版 中 完成 的 ， 如 果 读 者 使 用 的 是 其 他 版 本 的 Microsoft Windows 操 
作 系 统 ， 安 装 过 程 可 能 会 有 些许 差别 。 


2.2.1 安装 Apache HTTP 服务 器 


当 我 们 双击 在 上 一 节 中 获取 的 Apache HTTP 服务 器 安装 包 后 , Apache HTTP 服务 器 安 
装 与 配置 向 导 开 始 运 行 ， 如 图 2-11 所 示 。 单 击 “ 下 一 步 (Next) ”按钮 ， 可 以 看 到 Apache 
许可 证 授权 协议 ， 如 图 2-12 所 示 。 


‘Welcome tp the Installation Wizard for 


Lcense Agreement 
Apache HTTP Server 2.2.22 Please reac the folowng license agreement carefuly. 
The nstalabon Waerd wi netal Apoche HT Server 2.2.22 Apache Licen “~ 
Computer, To coniinue, dick Next Version 2.0, January 2004 国 
httpywrww apachaoranicensesi> 


TERMS AND CONDITIONS FOR USE REPRODUCTION, AND DISTRIBUTION 


1. Definitons, 
ee TLicenser shall mean the terms and conditions for use, reproduction, and 
pe distribution as defined by Sectons 1 through 9 ofths document 


not accept the terns in the icense agreement 


Iinstalshield 


Cm Let> ] 己 cnm 


图 2-11 Apache HTTP 服务 器 安装 向 导 首 页 图 2-12 接受 Apache 许可 证 协议 


选择 “我 接受 许可 证 授权 协议 中 的 条 款 (Iaccept the terms in the license agreement) ” 


然后 单 击 “ 下 一 步 (Next) ”按钮 ， 弹 出 “阅读 Apache HTTP 服务 器 简介 ”对 话 框 ， 如 图 
2-13 所 示 。 


Read Thts irst 
Reac ms efore Rurning Aoache on Wncows. 


Mpache HTTP Sever 
anataiz 国 
ne Apacme HrIF Senerts a poneny ananeaoe P's conoren ven sn 


Server mfonnebon 
iese erter yo server's rfermasen 


Networt Donain [e.g. omerct con) 


Sve hane (e.g. wm someret con); 


mmszsamubaemm 


atestversion 
is of ne alestversion can be found on he Apache HTTP sever prolectpage 


dal 
mp2yntpaapacha oral 


[ealshad 


rr er Eee 


图 2-13 ”Apache HTTP 服务 器 简介 图 2-14 设置 服务 器 信息 
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在 简介 中 ,可 以 了 解 Apache HTTP 服务 器 是 什么 及 如 何 获取 它 的 最 新 版 本 。 阅 读 完 成 
后 ， 单 击 “ 下 一 步 (Next) ”按钮 ， 开 始 设置 服务 器 信息 ， 如 图 2-14 所 示 。 

需要 设置 的 有 四 项 , 分 别 是 “网 络 域 (Network Domain)”、“ 服 务 器 名 (Server Name) ”、 
“管理 员 邮 件 地 址 (Administrator's Email Address) ”及 Apache HTTP 服务 器 提供 服务 的 端 
口号 ， 通 常情 况 下 保持 默认 即 可 。 若 当前 系统 中 已 经 安装 了 使 用 80 端口 的 应 用 程序 (如 
Microsoft IS) ， 请 选择 “只 为 当前 用 户 安 装 ， 使 用 8080 端口 (only for the Current User， 
on Port 8080, when started Manually.) ”。 

设置 完成 后 ， 单 击 “ 下 一 步 (Next) ”按钮 ， 选 择 安装 类 型 ， 如 图 2-15 所 示 。 此 时 ， 
请 选择 “ 自 定义 〈Custom) ”选项 ， 然 后 单 击 “ 下 一 步 ” (Next) 按钮 。 


Ey Apache HTP Rpache TTP 
Setup Type Castom Setup 
Choose the setup type that best suits your needs. ‘Select the program features you want installed. 
Please select a setup type, Clck on on icon in the listbelow to change how e feoture is installed, 
© Typical 
喝 Typical program features wil be installed. (Headers and Libraries 
for compling modues will not be installed,) 
custom te 
i a 2 
| 二 
hard dnve, 
a 
C:\Program Files\Apache Software Foundation Vipache2.2\ 
Em Next> | [Concal [ep |][ Space |][ <Bak [ Next> ]|[ Cancel |] 
2 安 站 选择 安 半 路 第 
图 2-15 确定 安装 类 型 图 2-16 选择 安装 路 径 


在 如 图 2-16 所 示 的 “ 自 定义 安装 (Custom Setup) ”界面 ， 单 击 “ 修 改 〈Change...) ” 
按钮 修改 安装 路 径 。 然 后 单 击 “ 下 一 步 (Next) ”按钮 ， 弹 出 “开始 安装 (Ready to Install 
the Program) ”对 话 框 ， 如 图 2-17 所 示 。 此 时 ， 单 击 “ 安 装 (Install) ”按钮 ， 安 装 程序 
便 会 按照 上 面 的 设置 将 Apache HTTP 服务 器 安装 到 指定 的 目录 里 。 


Ready to Install the Program 

The wizardis ready to begin installation, 

ptt bn We eto, The Instalation Wizard has successfuly nstaled Apache HTTP 
Sever 2.2. zard, 


1f you want to review or change any of your installation settings, dick Back. Cick Cancel to 22 Cick Finish to exitthe wi 
the wizrd， 


Instalishield 


Cm [me LL ce 


图 2-17 开始 安装 图 2-18 安装 结束 


当 弹 出 如 图 2-18 所 示 的 对 话 框 时 ， 整 个 安装 过 程 宣告 结束 。 在 执行 安装 时 ， 可 能 会 出 
现 一 些 命令 行 窗口 ， 无 视 即 可 。 单 击 “ 结 束 (Finish) ”按钮 ， 关 闭 对 话 框 。 
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2.2.2 ”安装 验证 


在 安装 结束 后 ， 读 者 可 以 通过 如 下 方法 来 验证 Apache HTTP 服务 器 安装 是 否 成 功 : 

口 检查 系统 托盘 中 是 否 出 现 了 辆 图 标 . 若 出 现 了 该 图 标 , 表示 Apache HTTP 服务 器 
安装 成 功 并 正在 运行 。 若 系统 托盘 中 出 现 的 是 图 图 奈 ， Apache HTTP 服务 器 安装 
成 功 ， 但 并 未 运行 ， 此 时 单 击 开始 菜单 ， 找 到 Apache HTTP 服务 器 的 文件 夹 。 然 
后 打开 “控制 Apache HTTP 服务 器 (Control Apache HTTP Server) ”文件 夹 ， 单 
击 “ 启 动 (Start) ”按钮 。 若 启动 成 功 ， 请 进行 下 一 步 的 验证 。 若 启动 失败 ， 则 
需要 调整 Apache 服务 器 提供 服务 的 端口 号 。 

口 打开 浏览 器 , 在 地 址 框 内 输入 http://localhost 并 按 回 车 键 。 若 出 现 如 图 2-19 所 示 的 


页 面 ， 表 示 Apache 服务 器 运行 正常 。 


(Ee 
Dm 


FT | 和 @ htpy/localhosy 从 


~ 回 呈 Pagev Safetyv Toosv 


It works! 


图 2-19 Apache 服务 器 安装 成 功 并 正常 运行 


2.2.3 配置 Apache HTTP 服务 器 


完成 Apache HTTP 服务 器 的 安装 之 后 ， 需 要 对 其 进行 简单 的 配置 ， 以 使 其 更 好 地 和 


PHP 引擎 互动 。 


单 击 屏幕 左下 角 的 开始 按钮 ， 单 击 “ 所 有 程序 (All Programs) ”|“Apache HTTP 服 
务 器 2.2 (Apache HTTP Server 2.2) ”|“ 配 置 Apache 服务 器 (Configure Apache Server) ” 
|“ 编 辑 Apache httpd.conf 配置 文件 (Edit the Apache httpd.conf Configuration File) ”选项 ， 


打开 httpd.conf 文件 ， 如 图 2-20 所 示 。 
在 文件 中 ， 需 要 找 如 下 几 行 代码 : 


ServerRoot "C:/PHP/Apache" 


Listen 80 


DocumentRoot "C:/PHP/Apache/htdocs" 


<Directory "C:/PHP/Apache/htdocs"> 


// 服 务 器 安装 目录 
// 服 务 器 监听 端口 
// 服 务 器 文件 目录 
// 服 务 器 文件 目录 
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图 2-20 ”Apache httpd.conf 文件 截图 


找到 这 些 代码 后 ， 请 确认 : 

口 服务 器 的 安装 目录 是 否 为 之 前 安装 Apache HTTP 服务 器 时 选择 的 目录 。 在 这 里 ， 
Apache HTTP 服务 器 路 径 为 C:/PHP/Apache。 

口 服务 器 的 监听 端口 是 否 为 80。 若 在 “2.2.2 ”安装 验证 ”小 节 中 发 现 Apache HTTP 
服务 器 无 法 正常 启动 ， 可 尝试 将 服务 器 的 监听 端口 改 成 8080 或 其 他 端口 ， 再 重启 
Apache HTTP 服务 器 。 

在 确认 过 Apache 服务 器 的 安装 目录 和 监听 端口 后 ， 需 要 修改 服务 器 的 文件 目录 。 在 

本 书 中 ， 服 务 器 的 文件 目录 的 路 径 为 C:/PHP/htdocs/。 修 改 后 的 httpd.conf 文件 如 下 : 


ServerRoot "C:/PHP/Apache" // 服 务 器 安装 目录 
nien 0 /7 服务 器 监听 端口 
en "C:/PHP/ htdocs" // 修 改 后 的 服务 器 文件 目录 
eDirectory "C:/PHP/ htdocs"> // 修 改 后 的 服务 器 文件 目录 


修改 完成 后 ， 请 重启 Apache HTTP 服务 器 。 
2.3 安装 和 配置 PHP 脚本 处 理 引 党 


由 于 获取 的 PHP 引擎 包 是 ZIP 格式 的 压缩 文件 ， 因 此 我 们 需要 先 将 其 解压 缩 , 然后 再 


2.3.1 解压 PHP 引擎 包 


如 图 2-21 所 示 , 右键 单 击 之 前 获取 的 PHP 引擎 包 文件 , 在 弹出 的 快捷 菜单 中 选择 “ 解 
压 (Extract All...) ”， 弹 出 如 图 2-22 所 示 的 对 话 框 。 
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php_547 Date modified 
yy Compressed bipped Folder | 


图 2-21 解压 PHP 引擎 包 


Select a Destination and Extract Files 


Files will be extracted to this folder: 
CApHP\PHP_Module 


园 Show extracted files when complete 


2-22 选择 解压 目标 路 径 


单 击 “ 浏 览 (Browse) ”按钮 ， 将 路 径 修改 为 C:/PHP/PHP_Module， 然 后 单 击 “ 解 压 
(Extract) ”按钮 。 解 压 完成 后 PHP Module 目录 自动 打开 并 显示 在 桌面 上 。 


2.3.2 配置 PHP 引擎 


在 使 月 


日 PHP 引擎 之 前 ， 我 们 需要 介绍 Apache 服务 器 和 PHP 引擎 互相 认识 。 


因 


此 ， 这 


部 分 的 配置 ， 我 们 分 两 步 走 。 第 一 步 ， 介 绍 PHP 引擎 给 Apache 服务 器 认识 ， 也 就 是 配置 
Apache 服务 器 的 httpd.conf 文件 ， 第 二 步 ， 介 绍 Apache 服务 器 给 PHP 引擎 认识 ， 也 就 是 
配置 PHP 引擎 的 php.ini 文件 。 通 过 这 两 个 步骤 ，Apache 和 PHP 引擎 就 互相 认识 了 对 方 并 
成 了 一 对 好 搭档 。 
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配置 Apache 服 务 器 的 httpd.conf 文 件 


打开 Apache 服务 器 的 httpd.conf 文件 。 找到 “Dynamic Shared Object (DSO) Support”， 


然后 在 一 连 串 的 LoadModule 后 加 上 一 行 : 


件 ， 


LoadModule php5 module "C:/PHP/PHP Module/php5apache2 2.d11" 


然后 找到 如 下 代码 : 
<IfModule dir module> 


DirectoryIndex index.html 
</IfModule> 


将 其 修改 为 : 


<IfModule dir module> 


DirectoryIndex index.html index.php // 将 index.php 设置 为 默认 首页 文件 
</IfModule> 


然后 找到 如 下 代码 : 


AddType application/x-gzip .tgz .gz 


在 其 后 添加 : 


AddType application/x-httpd-php.php 
AddType application/x-httpd-php.html 


后 保存 修改 好 的 httpd.conf 文件 。 
2. 配置 PHP 引 警 的 php.ini 文 件 


当 PHP 引擎 包 解压 完成 后 ， 我 们 会 在 C:/PHP/PHP Module 目录 下 发 现 两 个 php.ini 文 
-个 为 php.ini_dist, 另 一 个 为 php.ini recommanded。 这 里 我 们 将 后 者 重 命名 为 php.ini， 


然后 用 记事 本 打开 该 文件 ， 如 图 2-23 所 示 。 


File Edit Formet Vew Help 
[ED 


ary bests ,of .Pes beay or Fo far Pap.to 
aad Ie it nse Pornaped ee nd phe oaks Far Fe Fr Ene errome 
vering directory, in Che pEth desianated by the en abe 
:pnppc "oyng Tn eh pach Pha was deffned 1n onpf 1e Cne Cin chat ed。 
; Under wdndow?, hor eonod e-em pth $e tho ethene afrecrory Ro 
:pee A ohfeh che pres Pol le eTooked for "can be overr Toad ws 和 
en "fnP Opamand Hhe mad 


; The syntax of the file is extrenely simple. whitespace and Lines 

: beginning with a semicolon re 211 ly ignersd (2 You ereoably gassec- 
:Sesto eaders (, gp roo]> dps #20211ency torored, even rhough 
;they might mean sonething jn the future. 


: Mirectlves are, speciried using the following syntax: 
; directive ~ val 
: DFEEEVE aied Sre -case sensitive" - foo-bar 1s differen from Foo-bar。 


;hea de Can be a 2tLing, Pennber, nme constant (esos Ee, or Mepis on 
:of “the Th1 Constants (on’ off, True’ False, Yes, No and Nona) or an expréssion 
:Qe Be A BEG), Ors Mosted der ng CF): 


Expressions in the INI file are 1imited to bitwise operators and parentheses: 
| bitwise OR 
bitwise AND 
bitwise NOT 
boolean NOT 


; Boolean flaog, can berEur ned on, ssiog Ehe valyes #90" I"ve or ves. 
; They can be turned off using the values 0，off，F31se or 


; An empry string can be denoted by simply not writing anything after the equal 


图 2-23 PHP 引擎 的 php.ini 文件 
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在 文件 中 搜索 如 下 几 行 代码 : 
这 /7 服务 器 文件 根 目录 
demon oie S H/o // 扩 展 库 文件 目录 
Br i i // 会 话 暂 存 目 录 
然后 把 它们 修改 为 : 


doc root = " C:\PHP\htdocs"// 将 服务 器 文件 根 目录 设置 为 C:\PHP\htdocs 


extension dir = "C:\PHP\PHP Module\ext" 
// 将 扩展 库 文件 目录 设置 为 C:\PHP\PHP Module\ext 


session.save path = " c:\php\session temp" 


// 将 会 话 暂 存 目 录 设 置 为 c:\php\session temp 
接 下 来 ， 在 文件 中 搜索 如 下 几 行 代码 : 
iextension=php gd2.d1l //GD 库 的 扩展 文件 ， 用 于 在 线 生 成 、 修 改 图 片 
Brie ion=php gettext .dll //PHP 的 gettext () 函数 扩展 库 


FERGIE RE -dll // 多 字 节 字符 串 处 理 模 块 扩展 库 


| 可。 隐 //PHP 用 于 连接 MySQL 数据 库 的 扩展 库 
PE mysqli.dl1 //PHP 用 于 连接 MySQL 数据 库 的 扩展 库 
pr oy ils = //RPC 扩展 库 ， 启 用 后 ， 可 以 使 用 XML 传输 命令 和 数据 


代码 行 前 的 分 号 〈;) 表示 这 些 代 码 行 被 注释 掉 了 ， 也 就 是 说 服务 器 在 加 载 PHP 引擎 
时 ， 不 会 加 载 这 些 扩展 库 。 如 果 需 要 加 载 这 些 扩展 库 ， 就 需要 将 这 些 代 码 行 前 的 分 号 (;) 
去 掉 。 上 面 列 出 的 扩展 库 只 是 在 进行 PHP 脚本 开发 时 ， 经 常会 用 到 的 一 些 扩展 库 。 读 者 也 
可 以 根据 需要 ， 自 行 开启 其 他 的 扩展 库 。 

修改 完成 后 ， 保 存 该 php.ini 文件 ， 然 后 将 其 复制 到 C:\Windows 路 径 下 。 之 后 ， 再 将 
PHP_Module 目录 下 的 libmysql.dll 和 libmcrypt.dll 文件 复制 到 C:\Windows\System32 目录 
下 。 随 后 重启 Apache HTTP 服务 器 。 在 重启 完成 后 ， Apache 服务 器 就 算是 和 PHP 引擎 相 
互 认识 了 。 


2.3.3 配置 验证 


在 配置 完成 后 ， 需 要 验证 一 下 配置 是 否 正确 。 为 此 ， 需 要 先 编写 一 个 测试 文件 ， 并 将 
其 命名 为 index.php， 然 后 通过 浏览 器 访问 这 个 文件 。 

测试 文件 的 内 容 如 下 : 

<?php 

phpinfo(); 

> 


打开 记事 本 ， 将 上 述 代码 复制 到 记事 本 中 。 然 后 单 击 “文件 (File) ”| “另存 为 《Save 
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As...) ”按钮 ， 弹 出 如 图 2-24 所 示 的 对 话 框 。 


| Save As 
eS ss 


De 


Povo Name Date modified Type 


ml Desktop 
六 Downloads 
筷 Recent places 
E 


No items match your seareh. 


Libraries 
国 Documents 


oh Music 
图 Pictures 
国 videos 


Computer = 
File name: 四 | 
Saveastype [Text Documents (bd) ~ 


Tet Documents Ey 


eS Hide Folders 


图 2-24 将 文件 保存 为 index.php 


在 “文件 类 型 (Save As type) ”下 拉 框 中 单 击 “ 所 有 文件 (All Files) ”按钮 ， 在 “ 文 
件 名 (File name) ”文本 框 中 填写 index.php， 然 后 单 击 “ 保 存 ” 按 钮 ， 将 文件 保存 在 服务 
器 文件 根 目录 (C:\PHP\htdocs) 里 。 

打开 Internet Explorer 浏览 器 ， 在 地 址 栏 中 输入 http://localhost， 然 后 回 车 查看 结果 
如 果 浏 览 器 显示 如 图 2-25 所 示 的 页 面 ， 说 明 PHP 配置 成 功 。 


ro mr 


SS 一 一 一 


CE = | 


nows NT WF IHOTOAMCPU 6 1 bu 7600 
en 52011 17.2008 

sciot malooe confioure js “—enatbie-snaps hot-buid enabledebuDacr “wy 
|anaoehottamclate=dione sdcanep_S_2Nc6vedtamclats "wi ph: 

| en 5-Zucmaoene. Put Mh Pap- p00 ore seceraaewnetonclenelD 
td 


FE 


@ intema| Pretected Mede Off 


图 2-25 PHP 版 本 信息 
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2.4 安装 和 配置 MySQL 数据 库 


在 完成 Apache 服务 器 和 PHP 引擎 的 安装 与 配置 之 后 ， 需 要 继续 安装 和 配置 MySQL 
数据 库 ， 以 便 为 正在 搭建 的 PHP 平台 提供 强大 的 数据 库 支 持 。 需 要 注意 的 是 ，MySQL 数 
据 库 的 安装 需要 有 Microsoft .NET 4.0 支持 。 若 读者 的 操作 系统 中 没有 安装 NET 4.0， 请 到 
微软 的 官方 网 站 下 载 。 


2.4.1 安装 MySQL 数据 库 


双击 在 2.1.3 小 节 中 获取 的 MySQL 数据 库 安装 包 ， 弹 出 如 图 2-26 所 示 的 对 话 框 。 


图 2-26 ”MySQL 数据 库 安装 界面 
单 击 “ 安 装 MySQL 产品 (Install MySQL Products) ”链接 后 的 效果 如 图 2-27 所 示 。 
选中 “我 接受 这 份 许可 证 协议 中 的 条 款 〈Iaccept the license terms) ”选项 ， 然 后 单 击 “ 下 
一 步 (Next) ”按钮 。 


2-27 MySQL 许可 证 协议 
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因为 下 载 的 是 最 新 稳定 版 本 的 MySQL 软件 ， 因 此 在 随后 出 现 的 “获取 软件 最 新 更 新 
(Find latestproducts) ”页 面 中 ， 如 图 2-28 所 示 ， 选 中 “ 跳 过 更 新 检查 〈Skip the check for 
updates) ”选项 ， 然 后 单 击 “ 执 行 (Execute) ”按钮 。 


Find latest products 


RN 
MySQL. Installer 


Before the nstallation is performed the Instaber will check if there are newer 
versions of the products You are about to instal / already instabed are avaiable. 


Medlatestproducts 


图 2-28 ”获取 软件 最 新 更 新 


更 新 检查 分 为 两 步 ， 第 一 步 将 会 检查 Intemet 连接 ， 若 检查 通过 ， 则 会 获取 产品 更 新 
信息 。 读 者 也 可 以 直接 单 击 “ 执 行 (Execute) ”按钮 来 获取 最 新 更 新 。 

在 接 下 来 的 页 面 里 ， 需 要 选择 安装 类 型 。 如 图 2-29 所 示 ， 一 共有 五 种 安装 类 型 。 它 们 
分 别 是 开发 者 〈 默 认 ) 、 仅 服务 器 、 仅 客户 端 、 完 全 安装 和 自 定义 。 


Choosing a Setup Type 


Please select the Setup Type that suits your use case. 


图 2-29 选择 安装 类 型 


口 选择 开发 者 〈 默 认 ) ， 将 会 安装 进行 MySQL 相关 开发 所 需 的 文件 。 
口 选择 仅 服务 器 ， 将 且 仅 会 安装 MySQL 服务 器 程序 。 


“Ns 
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口 选择 仅 客户 端 ， 将 且 仅 会 安装 MySQL 客户 端 程序 。 
口 选择 完全 安装 ， 将 会 安装 MySQL 服务 器 和 客户 端 程序 。 

口 选择 自 定义 安装 ， 读 者 可 以 根据 自己 的 需要 选择 安装 的 内 容 。 

按照 之 前 的 规划 ，phpMyAdmin 会 被 用 来 管理 MySQL 数据 库 ， 因 此 ， 这 里 选择 的 安 
装 方式 为 仅 服务 器 。 然 后 将 “安装 路 径 〈Installation Path) ”和 “数据 路 径 (Data Path) ” 
分 别 修改 为 C\PHPIMySQL 和 C:\PHPMySQL\IMySQL Server 5.5\， 然 后 单 击 “ 下 一 步 
(Next) ”按钮 。 
在 执行 安装 之 前 ， 系 统 会 列 出 需要 安装 的 项 目 ， 如 图 2-30 所 示 。 


Installation Progress 


The fobowing products wa be nstaliod or updated. 


图 2-30 需要 安装 的 项 目 


安装 项 目 确认 无 误 后 ， 单 击 “ 执 行 (Execute) ”按钮 开始 安装 。 安 装 完成 后 ， 确 认 列 
表 中 各 安装 项 目的 “状态 (Status) ” 栏 均 显示 为 “安装 成 功 (Install success) ”， 然 后 单 
击 两 次 “下 一 步 (Next) ”按钮 ， 进 行 MySQL 服务 器 的 配置 界面 ， 如 图 2-31 所 示 。 


Mo ma 
r 


RN LServer Configurati 
MySQL. Installer ens eek 


图 2-31 配置 MySQL 服务 器 
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保持 默认 配置 即 可 。 直 接 单 击 “ 下 一 步 (Next) ”按钮 。 在 随后 弹出 如 图 2-32 所 示 的 
界面 上 ， 可 以 设置 root 账户 的 密码 和 添加 新 的 数据 库 用 户 。 在 这 里 ， 建 议 只 设置 root 账户 
密码 ， 不 添加 新 的 数据 库 用 户 。 


MySQL Server Configuration 
MySQL. Installer 


Root Account Password 


Enter the password for the root acrount Please remember io store 
this Possword no scare piece 
MySQt Root Password: | 外 


图 2-32 设置 root 账户 的 密码 和 添加 新 的 数据 库 用 户 


设置 完成 后 ， 单 击 “ 下 一 步 (Next) ”按钮 ， 弹 出 如 图 2-33 所 示 的 界面 。 在 此 ， 读 者 
可 以 设置 MySQL 的 服务 名 称 。 推 荐 保持 默认 设置 即 可 。 


N 


MySQL. Installer 


Windows service Name: 
园 start the MySQL Server at System Startup 


图 2-33 设置 MySQL 服务 名 


单 击 “ 下 一 步 (Next) ”按钮 ， 结 束 MySQL 服务 器 的 配置 。 
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2.4.2 配置 验证 


当 完成 了 上 面 的 配置 之 后 ， 还 得 需要 验证 一 下 配置 是 否 正确 。 

首先 ， 单 击 开始 菜单 ， 选 择 “ 所 有 程序 (All Programs) ”|“MySQL”|“MySQL Server 
5.5” 命 令 ， 弹 出 如 图 2-34 所 示 的 MySQL 5.5 命令 行 客户 端 (MySQL 5.5 Command Line 
Client) 对话 框 。 


国 MysQL55 Command Lne 0 NE、v- 2 = 


图 2-34 MySQL 5.5 命令 行 客户 端 


按照 系统 提示 ， 输 入 之 前 设置 的 密码 ， 然 后 回 车 ， 进 入 MySQL 5.5 系统 提示 符 界面 ， 
如 图 2-35 所 示 。 


国 My5QL55 Commond Une Cherl 


图 2-35 MySQL 5.5 系统 命令 提示 符 


在 系统 提示 符 界面 ,可 以 看 到 诸如 MySQL 连接 号 、MySQL 服务 器 版 本 、 版 权 信息 及 
命令 行 帮助 信息 。 若 需要 通过 命令 行 操作 MySQL 数据 库 软 件 ， 请 参阅 MySQL 服务 器 的 
口 


产品 文档 。 
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2.5 安装 和 配置 phpMyAdmin 


其 实 当 安 装 和 配置 过 程 进行 到 这 里 ， 就 可 以 收工 了 ， 因 为 Apache HTTP 服务 器 +PHP 
脚本 处 理 引擎 +MySQL 数据 库 的 环境 就 算是 搭建 起 来 了 。 换 句 话说， 如 果 读 者 知道 如 何 通 
过 命令 行 来 管理 数据 库 ， 那 么 这 一 节 的 内 容 就 可 以 忽略 掉 了 。 

但 是 ， 读 者 也 同样 错过 了 一 个 认识 到 PHP 强大 能 力 的 机 会 。 在 获取 phpMyAdmin 的 
时 候 , 我 们 曾经 介绍 过 , phpMyAdmin 是 一 款 完 全 由 PHP 脚本 写 的 数据 库 管理 软件 。 因此， 
phpMyAdmin 的 源 代 码 是 认识 并 学 习 PHP 和 数据 库 交 互 的 最 好 教材 。 那 么 闲话 少 说 ， 现 在 
就 来 了 解 一 下 如 何 安装 和 配置 这 款 基于 Web 的 MySQL 数据 库 管理 软件 吧 。 


2.5.1 解压 phpMyAdmin 压缩 包 


在 之 前 获取 的 pppMyAdmin 压缩 包 上 右键 单 击 ， 选 择 “ 解 压 所 有 (Extract All...) ” 
命令 。 弹 出 如 图 2-36 所 示 的 对 话 框 。 单 击 “ 浏 览 (Browse) ”按钮 修改 解压 路 径 为 
C:\PHP\htdocs\phpMyAdmin3， 然 后 单 击 “ 解 压 (Extract) ”按钮 。 


过 二 baract Compressed (Zipped) Folders 


Select a Destination and Extract Files 


Files will be edracted to this folder: 
CNPHP\htdocs\phpMyAdmin3 Browse 


司 Show extracted files when complete 


图 2-36 解压 phpMyAdmin 讨 缩 包 到 指定 目录 


解压 完成 后 ，phpMyAdmin 目录 自动 打开 并 显示 在 桌面 上 。 这 样 ， 就 算是 完成 了 
phpMyAdmin 压缩 包 的 解压 工作 。 


2.5.2 配置 phpMyAdmin 


在 浏览 器 地 址 栏 中 输入 http://localhost/phpmyadmin3/setup， 然 后 按键 盘 上 的 回 车 键 ， 
会 弹出 如 图 2-37 所 示 的 配置 界面 。 
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合 - 上 - 口 二 - paoe- sy Tock- 着” 


I 
ET | 
概要 


于 元 攀 巷 信息 2 
服务 器 


商 隶 服务 器 
杞 置 文件 
默认 请 言 外国 
苇 认 服务 器 @ 
换行 符 


nprAden 主页 (失信, 至 女 ) 侣 由 他 ， 六 六 ) 过 


图 2-37 phpMyAdmin 安装 配置 界面 


(1) 根据 “服务 器 ”区 域 框 内 的 提示 信息 显示 ,系统 中 没有 配置 好 的 服务 器 


。 单 击 “ 新 
建 服务 器 ”按钮 ， 在 弹出 如 图 2-38 所 示 的 页 面 中 新 建 一 个 服务 器 。 


phpMyAdmin setup 
添加 服务 器 
基本 设 着 ”认证 | 服务 器 设置 “高 级 功能 | 始 改 泥 啼 


图 2-38 添加 服务 器 


(2) 直接 单 击 该 页 面 下 方 的 “保存 ”按钮 ， 系 统 会 返回 到 安装 首页 ， 并 弹出 如 图 2-39 
所 示 的 提示 信息 。 

按照 系统 的 提示 ， 可 以 使 用 SSL 来 加 密 与 数据 库 服 务 器 的 连接 。 不 过 在 本 书 中 ， 数 据 
库 服务 器 就 是 本 地 计算 机 ， 可 以 不 使 用 SSL 连接 。 另 外 ， 系 统 自 动 为 我 们 设置 了 一 个 短语 
密码 用 于 Cookies 加 密 。 到 此 ， 服 务 器 就 添加 完成 了 。 

(3) 在 提示 信息 下 方 的 “服务 器 ”区 域 框 中 ， 可 以 看 到 已 经 添加 的 服务 器 相关 信息 。 


在 “配置 文件 ”区 域 框 中 ， 可 以 看 到 需要 保存 的 配置 信息 。 确 认 无 误 后 ， 单 击 “ 保 存 ” 按 
钮 ， 如 图 2-40 所 示 。 


ms 
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| 国 使 用 SSL (localhost [1]) 
各 虹 吉 所 车 服务 基 支 持 ， 售 字 交 合用 SSL 汪汪- 


| 日 得 二 志 到 


各 打开 了 Coelies 认证 方式 ， 人 二 有 议 要 本 主讲 三， 因此 系统 自动 生 戒 了 一 个 二 训 宣 矶 ,这 
用 于 加 窗 Cookies- 您 不 天 要 全 这 个 拓 话 生 码 。 


图 2-39 ”完成 添加 服务 器 后 出 现 的 系统 提示 信息 


服务 器 

| 

| 名字 认证 方式 DsN 

| 1 locahost conke mysqli ocalbowt 时 要 出院 

| 画 丁 到 

| 配置 文件 

| 

| 默认 语言 四 四 Ergish 回 
| 紧 认 服务 器 @@ [cr 

| 换行 符 windows (on) 加 


| 加 Em 


图 2-40 已 添加 的 服务 器 相关 信息 和 需要 保存 的 配置 文件 信息 
(4) 当 系 统 弹出 如 图 2-41 所 示 的 提示 时 ， 配 置 文件 保存 成 功 。 在 按照 提示 信息 将 配置 
文件 从 config 文件 夹 中 复制 到 phpMyAdmin 的 根 目录 之 后 , 就 可 以 使 用 phpMyAdmin 来 管 
理 MySQL 数据 库 了 。 


国 配置 已 保存 。 
| 了 轩 已 保 到 configconfig nc php 文件 ， 请 将 其 复制 到 phpMyAdmi 根 目录 并 删 际 config 文件 


2-41 配置 文件 保存 成 功 


在 浏览 器 中 输入 http://localhost/phpmyadmin3, 然后 按键 盘 上 的 回 车 键 , 弹出 如 图 2-42 
所 示 的 登录 界面 。 这 时 ， 可 以 使 用 root 用 户 登录 phpMyAdmin 系统 。 有 同学 可 能 会 问 ， 密 
码 是 什么 啊 ? 密码 就 是 你 在 配置 MySQL 服务 器 时 设置 的 那个 MySQL Root Password。 不 
记得 的 同学 面壁 吧 。 


phpMUAdmin 
帮凶 使 用 phpMyAdmin 


图 2-42 ” phpMyAdmin 登录 页 面 
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2.6 使 用 套件 包 搭 建 PHP 运行 环境 


有 的 同学 看 到 这 里 ， 可 能 已 经 头 昏 眼花 、 汗 流 淡 背 了 。 有 的 同学 甚至 会 感叹 道 : “ 拱 
建 个 环境 都 这 么 复杂 ， 那 正 儿 八 经 地 学 起 PHP 脚本 语言 来 会 不 会 更 难 啊 ? ” 

为 了 提高 大 家 搭建 PHP 运行 环境 的 效率 ， 有 很 多 的 大 牛 开 发 了 不 同 版 本 的 一 键 安装 
包 ， 而 且 这 些 安装 包 差不多 都 是 免费 的 。 本 书 只 介绍 两 个 : 一 个 为 国产 软件 PHPnow， 另 
一 个 为 常用 的 WampServer。 大 家 可 以 按照 个 人 的 需要 进行 选择 。 需 要 注意 的 是 ， 两 者 都 
只 能 安装 在 32 位 的 Windows 操作 系统 中 。 


2.6.1 PHPnow 


按照 PHPnow 官方 网 站 的 介绍 , PHPnow 是 一 款 Windows 32 位 操作 系统 下 绿色 、 免费 
的 Apache+PHP+MySQL 环境 套件 包 。 使 用 该 套件 包 可 以 方便 快捷 地 搭建 支持 虚拟 主机 的 
PHP 环境 。 套 件 包 附 带 PnCp.cmd 控制 面板 ， 帮 助 用 户 快速 地 配置 套件 ， 使 用 非常 方便 。 

在 这 一 小 节 里 ， 我 们 就 看 看 如 何 下 载 、 安 装 和 配置 PHPnow 套件 包 。 

(1) 访问 PHPnow 官方 网 站 (http:/www.phpnow.org) ， 下 载 PHPnow 的 最 新 套件 包 。 
解压 后 双击 如 图 2-43 所 示 的 setup.cmd 文件 。 


[ecm RCI 


© rr BE Er 


Organize = Includeinlibray = Sharewithv Bum Newfolder 


六 Favorites ee 


Desdtop 图 7zdll 

及 Downloads 回 7z 

通 Recent Places [DD Package7z 

DD) Readme 

局 Lbraries 园 Setup 

国 Documents 丫 可 |B 志 

oh Music 癌 关于 种 坟 

国 Pictures 口 升级 方法 

加 videos Run as administrator 


Troubleshoot compatibilty 
i Computer Restore previous versions 


Open 


Print 


Sendto 
全 Network 
Cut 


图 2-43 ”安装 PHPnow 套件 包 


(2) 根据 系统 提示 ， 我 们 需要 选择 Apache 服务 器 版 本 和 MySQL 服务 器 版 本 系统 ， 如 
图 2-44 所 示 。 在 输入 所 需 版 本 后 ， 系 统 会 自动 将 所 需 文件 解压 到 当前 目录 。 
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图 2-44 
(3) 在 解压 完成 后 , 系统 会 要 求 
2-46 所 示 。 


择 Apache 和 MySQL 版 本 
系统 将 启动 Apache 服务 和 MySQL 服务， 并 要 tt 


执行 Init.cmd 进行 初始 化 ， 


|: 


如 图 2-45 所 示 。 输 入 立 后， 
村 MySQL 服务 器 root 用 户 的 密码 ， 如 图 


mt 


图 2-45 
(4) 设置 密 
图 


h 


执行 Init.cmd 初始 化 


图 2-46 设置 root 用 户 密 胡 
码 后 ， 系 统 提示 “全 部 完成 ! ! 你 将 可 以 看 到 PHPnow 的 默认 页 面 ! ”如 
2-47 所 示 。 按 任意 键 关闭 当前 窗口 。 此 时 可 以 打开 浏览 器 ， 并 在 地 址 栏 中 输入 
ttp://localhost 来 验证 Apache+PHP+MySQL 的 环境 是 否 搭 建成 功 。 如 果 在 浏览 器 中 看 到 如 
图 2-48 所 示 的 页 面 时 ， 说 明 环境 搭建 成 功 。 


图 2-47 完成 设置 


图 2-48 PHPnow 安装 成 功 
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(5) 在 此 页 面 上 ， 在 “MySQL 连接 测试 ”区 域 框 内 的 “MySQL 用 户 密码 ”文本 框 内 
输入 之 前 设置 的 root 用 户 密码 ， 就 可 以 测试 MySQL 连接 。 若 测试 成 功 ， 页 面 显 示 结 果 如 
图 2-49 所 示 。 


MySQL 到 试 结果 
服务 器 localhost OK (5.0.90-community-nt) 
教 据 库 test OK 


图 2-49 MySQL 连接 测试 结果 


(6) PHPnow 套件 包 还 提供 了 MySQL 数据 库 在 线 管 理 软件 pppMyAdmin。 读 者 只 需 
单 击 “ 服 务 器 信息 〈Server Information) ”区 域 框 中 的 相应 链接 即 可 。 

当 确 认 PHPnow 安装 配置 成 功 后， 可 以 看 到 现在 的 phpnow 文件 夹 的 内 容 与 之 前 相 比 
发 生 了 变化 。 一 起 来 认识 一 下 如 图 2-50 所 示 的 文件 和 文件 夹 。 

在 这 个 文件 夹 中 : 

口 文件 PnCp.cmd 是 PHPnow 的 控制 面板 ， 如 图 2-51 所 示 。 双 击 该 文件 可 以 打开 
PHPnow 控制 面板 , 进而 完成 诸如 新 建 、 修改 和 删除 虚拟 主机 之 类 的 配置 服务 器 的 
操作 。 

口 文件 夹 htdocs 是 PHPnow 用 来 存放 网 页 文件 的 服务 器 根 日 录 。 可 以 将 所 有 的 网 页 
文件 放 在 这 个 目录 中 ， 并 通过 http://localhost 地 址 来 访问 它们 。 

口 文件 夹 Apache-20、MySQL-5.0.90 和 php-5.2.14-Win32 包含 了 Apache 服务 器 、 
MySQL 服务 器 和 PHP 引擎 的 相关 文件 。 

口 文件 夹 Pn 和 PnCmds 则 包含 了 PHPnow 的 相关 文件 和 命令 。 


四 Mim tsi Prow 55 


PHPnow 控 制 面板 


图 2-530 phpnow 文件 夹 图 2-51 PHPnow 控制 面板 


2.6.2 WampServer 


WampServer 包含 的 组 件 有 : Apache 服务 器 、MySQL 服务 器 、PHP 处 理 引 擎 、 
phpMyAdmin 数据 库 管 理 软件 和 Xdebug 组 件 。 
在 这 一 节 里 我 们 来 看 看 如 何 下 载 、 安 装 和 配置 WampServer。 


Cp 
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(1) 访问 WampServer 官方 网 站 ， 下 载 WampServer 的 安装 包 。 下 载 完成 后 ， 双 击 安 
装 包 文本 ， 开 始 安装 和 配置 过 程 ， 如 图 2-52 所 示 。 单 击 “ 下 一 步 (Next) 按钮 ”。 

(2) 在 如 图 2-53 所 示 的 页 面 中 指定 WampServer 的 安装 路 径 ,然后 单 击 “ 下 一 步 (Next)” 
按钮 。 安装 完成 过 程 中 , 系统 会 要 求 读者 指定 开发 过 程 中 需要 使 用 的 浏览 器 , 直接 单 击 “ 打 
开 (Open) ”按钮 将 使 用 Intemet Explorer 作为 默认 浏览 


Setup - WampSever Setup WampServer : 
A Welcome to the WampServer 2 ee ee 人 
WwW | Where should WampServer 2 be instaled; 
a sd a 
Itis recommended that you dcse al other applications before 
Ponired by ne To continye, cidk Next. If you woud lke to select a different folder, cick Browse, 
er Way st et Seup. me 
een, ttn cra oases ms Tew] 
Open Source 
Service Provider 
http:/www.alterway.fr 
MyS0L :5.5.24 
和 an :3.01 
eg #2.2,0 At aact 237.3 MB of fee dick space is required, 
Goneel Sok | [Con 
ey So 
图 2-52 WampServer 安装 界面 图 2-53 指定 WampServer 的 安装 路 径 


(3) 安装 完成 后 ， 检 查 系 统 托盘 中 的 国 图 标 。 若 图 标 为 绿色 ， 表 示 服 务 器 启动 成 功 ， 
为 橙色 ， 服 务 器 启动 失败 ， 请 安装 VC++ 2010 运行 库 后 再 试 。 
(4) 在 确认 服务 器 启动 成 功 后 , 单 击 系统 托盘 中 的 国 , 弹出 如 图 2-54 所 示 的 快捷 菜单 。 


a 

5 四 
图 生生 生生 
se < 二 4 用 
:Ee 四 二 三 站 | 
人 SE 
| 5 ele < ea 
“ eleprle 


CT YINMNYISAdWVM 


图 2-54 WampServer 快捷 菜单 


通过 此 快捷 菜单 ,可 以 根据 自己 的 需要 定制 Apache 服务 器 、 PHP 引擎 和 MySQL 服务 
器 的 相关 设置 。 这 里 需要 注意 的 是 ，MySQL 服务 器 的 root 用 户 密码 为 空 。 若 有 设置 root 
用 户 密码 的 需要 ， 请 选择 “MySQL” | “my ini” 命令， 在 弹出 的 文本 文件 中 找到 如 下 儿 行 
代码 ; 


# The following options will be passed to all MySQL clients 


[client] 

#password = your password 
port = 3306 

socket = /tmp/mysql.sock 


然后 将 元 assword 一 行 前 的 # 号 去 掉 ， 然 后 在 该 行 的 等 号 后 面 输入 需要 的 密码 。 确 认 无 


三 


第 2 章 搭建 PHP 运行 环境 


误 后 ， 保 存 并 关闭 文件 ， 最 后 重启 MySQL 服务 即 可 。 
2.7 在 微软 IIS 上 配置 PHP 运行 环境 


在 第 一 章 里 ， 我 们 还 提 到 PHP 除了 可 以 和 Apache 的 HTTP 服务 器 搭档 之 外 ， 也 可 以 
与 Windows 操作 系统 自 带 的 微软 IIS 服务 器 合作 。 本 节 将 介绍 如 何 让 微软 IS 服务 器 认 
识 PHP。 


2.7.1 开启 互联 网 信息 服务 


首先 ， 我 们 需要 知道 的 是 ， 在 Windows 操作 系统 中 自 带 有 一 个 Web 服务 ， 名 字 叫 做 
互联 网 信息 服务 (Intemet Information Services) 。 但 是 ， 这 一 服务 除了 在 服务 器 版 的 操作 
系统 中 是 默认 开启 的 以 外 ， 在 其 他 版 本 的 Windows 操作 系统 中 ， 都 是 默认 关闭 的 。 因 此 ， 
第 一 步 ， 需 要 启用 互联 网 信息 服务 。 

打开 “控制 面板 ”， 找 到 并 双击 “程序 和 功能 ”图 标 ， 在 弹出 的 窗口 左 侧 找到 “开启 
或 关闭 Windows 功能 ”。 然 后， 在 弹出 的 “Windows 功能 (Windows feature) ”对 话 框 中 
找到 “互联 网 信息 服务 (Intemet Information Services) ”， 单 击 其 前 方 的 复 选 框 所 示 。 然 
后 展开 “互联 网 信息 服务 ”节点 ， 找 到 并 展开 “万 维 网 服务 (World Wide Web Services) ” 
节点 ， 找 到 并 展开 “应 用 开发 功能 (Application Development Features) ”节点 ， 接 着 勾 选 
该 节点 下 的 所 有 子 项 ， 如 图 2-55 所 示 。 


加 Windows Features [ele 


Turn Windows features on or off © 


To turn a feature on select its check box To turn a feature of clear its 
check box. A filled box means that only Part of the feature is tuned on, 


昌国 区 
一 ] Indeine gr 


田 国 膨 Web Management Tools 
日 国有 World Wide Web Sevices 
日 问 B Application Development Features 
贺电 .NET Edensibilty 


图 2-55 开启 互联 网 信息 服务 (Internet Information Services) 


在 单 击 “ 确 定 ” 按 钮 关闭 上 述 对 话 框 后 , 回 到 控制 面板 。 双击 “管理 工具 (Administrative 
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Tools) ”图 标 ， 看 看 里 面 是 否 存在 “互联 网 信息 服务 管理 器 (Internet Information Services 
(IIS) Manager) ”的 快捷 方式 。 若 存在 ， 表示 IIS 服务 已 经 成 功 启 用 了 。 双 击 打开 互联 网 信 
息 服务 管理 器 ， 会 弹出 如 图 2-56 所 示 的 窗口 。 


| 全 peeutweb site Home 


-请 c= - 风 Show Ml jcoopbr wm 


图 2-56 互联 网 信息 服务 管理 器 主 界面 


在 此 窗口 中 ， 可 以 看 到 很 多 的 选项 ， 不 过 大 部 分 跟 今 天 的 任务 无 关 。 我 们 只 需要 展开 
窗口 左 侧 导航 树 中 的 “站 点 〈Sites) ”， 选 中 “默认 站 点 (Default Web Site) ”。 然 后 单 
击 窗口 右 侧 的 “浏览 站 点 (Browser Web Site) ”区 域 框 内 的 “浏览 *:80 (http) ”链接 即 
可 。 在 单 击 了 该 链接 后 ， 可 以 看 到 如 图 2-57 所 示 的 页 面 ， 表 明 IIS 服务 成 功 启用 了 。 


图 2-57 微软 IS 服务 欢迎 页 面 
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2.7.2 ”为 微软 IIS 服务 添加 PHP 支持 


由 于 微软 HS 服务 对 PHP 不 提供 原生 支持 ， 所 以 需要 为 其 添加 PHP 支持 。 首 先 ， 需 要 
下 载 PHP 引擎 包 ， 这 一 点 与 2.3 节 中 的 内 容 类 似 ， 我 们 就 不 多 说 了 。 将 下 载 解压 好 的 PHP 
引擎 包 放 在 一 个 容易 找到 的 路 径 下 ， 然 后 打开 “php.ini _ recommended” 文件 ， 将 其 重 命名 
为 “php.ini”， 按 照 2.3 节 中 的 方法 修改 “php.ini” 文 件 。 然 后 将 修改 好 的 “php.ini” 文 件 
放 到 “C:\Windows\” 路 径 下 。 

做 好 这 项 工作 后 , 需要 打开 IIS 管理 器 , 单 击 管理 器 左 侧 导 航 树 的 顶级 节点 (如 图 2-56 


所 示 的 “TestServer” 节 点 ) 。 然 后 在 管理 器 的 中 部 区 域 框 中 找到 并 双击 “处 理 程序 映射 
(Handler Mappings) ”图 标 ， 进 入 “处 理 程序 映射 ”区 域 枉 ， 如 图 2-58 所 示 。 
i Testserver Home i Handler Mappings 
Fr 多 62 - 司 sewA jampar ww - 层 - et forteto gpealy he reeeurees soch ae DLbe nd marge code that ondereeponiee 
1 CR 
Comection Machinekey pagesand 。 Proydes Session State SMTP E-mail en 
et ee i 总 es pr 
ss De 
这 
pe 本 
Se a ER 区 
园 “ 芭 Aaron ore En 
on en 
国 训 斌 及 车 后 | HappemotngHanderfaaoqy .~seap Enabled Hinepecifind 
Lad Elie Mot Ome ht se ED 和 abs。 pealed 
pe ee em ett Corre 
egordefoctorr earded “op as。 
名 i 
a es 
Ss 


图 2-58 ”处 理 程序 映射 区 域 框 


在 “处 理 程序 映射 ”区 域 框 的 空白 处 , 单 击 右键 弹出 快捷 菜单 。 在 快捷 菜单 中 选择 “ 添 
加 模块 映射 《Add Module Mapping) ”命令 ， 弹 出 “添加 模块 映射 ”对 话 框 ， 如 图 2-59 
所 示 。 

按照 图 2-59 所 示 填 写 好 各 项 参数 。 需 要 注意 的 是 ，“ 可 执行 文件 (Executable) ” 参 
数 需 要 指定 “php_cgiexe” 文 件 的 路 径 ， 读 者 可 以 在 PHP 引擎 包 解 压 后 的 路 径 中 找到 它 。 
填写 好 各 项 参数 后 ， 单 击 “ 确 定 (OK) ”按钮 ， 关 闭 对 话 框 并 回 到 “处 理 程序 映射 ”区 域 
框 。 你 会 发 现 ， 刚 才 添 加 的 处 理 程序 映射 出 现在 了 在 区 域 框 中 。 

接着 ， 还 需要 为 PHP 应 用 指定 默认 文件 。 

打开 IIS 管理 器 ， 选 中 左 侧 导 航 树 的 顶级 节点 。 然 后 双击 管理 员 中 部 区 域 框 中 的 “ 默 
认 文 件 (Default Document) ”图 标 ， 进 入 “默认 文件 ”区 域 框 。 

在 该 区 域 框 的 空白 处 单 击 右键 。 在 弹出 的 快捷 菜单 中 选择 “添加 (Add...) ”命令 ， 
弹出 “添加 默认 文件 ”对 话 框 。 在 对 话 框 中 输入 “index.php” 后 ， 单 击 “ 确 定 (OK) ” 按 
钮 ， 关 闭 对 话 框 。 

至 此 ， 微 软 IS 和 PHP 处 理 引 擎 就 算是 认识 了 。 
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CNphp\php-cgiexe 


| Name: 
| PHpl 


| [rr 


图 2-59 添加 模块 映射 
2.7.3 ”验证 微软 IS 服务 对 PHP 的 支持 


现在 来 验证 一 下 微软 IS 服务 对 PHP 的 支持 情况 。 首 先 在 左 侧 导航 栏 的 “站 点 ”节点 
上 右键 单 击 ， 弹 出 快捷 菜单 。 在 快捷 菜单 中 ， 选 择 “ 新 建站 点 〈Add New Site) ”命令 ， 
弹出 “新 建站 点 ”对 话 框 ， 如 图 2-60 所 示 。 


Ste name: 
testphp 


Content Directory 
Physical path: 
Cinetpub\wwwroot\testphp 


Example: www.contoso.com or marketing.contoso.com 


ED Start Web site immediately 


图 2-60 添加 新 站 点 


在 填写 好 各 项 参数 后 ， 单 击 “ 确 定 (OK) ”按钮 ， 关 闭 对 话 框 。 这 时 ， 刚 才 添 加 的 站 
点 就 出 现在 了 左 侧 导 航 树 的 “站 点 ”节点 下 。 
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接 下 来 需要 在 站 点 的 根 目录 下 添加 一 个 名 为 “index.php” 的 文件 ， 然 后 在 文件 中 输入 
如 下 代码 : 


<?php 
phpinfo(); 
> 


代码 输入 完毕 后 ， 保 存 该 文件 。 然 后 回 到 IIS 管理 器 ， 选 中 “testphp” 节 点 ， 然 后 单 
击 管理 器 窗口 右 侧 的 “浏览 站 点 ”区 域 框 中 的 链接 ， 即 可 在 浏览 器 中 查看 该 网 站 了 ， 如 图 
2-61 所 示 。 


图 2-61 查看 “testphp” 站 点 


2.8 安装 集成 开发 环境 ( IDE ) 


工 欲 善 其 事 ， 必 先 利 其 器 。 在 搭建 完成 PHP 运行 环境 后 ， 还 需要 一 个 集成 开发 环境 用 
于 编写 和 调试 PHP 脚本 。 虽 然 说 用 文本 编辑 器 也 可 以 完成 PHP 脚本 的 开发 和 调试 ， 但 是 
使 用 文本 编辑 器 毕竟 没有 集成 开发 环境 来 得 快 ， 同 时 也 会 增加 PHP 学 习 曲 线 的 陡峭 程度 。 
一 旦 在 使 用 文本 编辑 器 编写 脚本 出 错 调 试 无 果 时 ,很 容易 使 PHP 新 手 产生 朋 难 的 情绪 ， 导 
致 放弃 学 习 。 

说 起 来 ,， PHP 真 的 不 难 ， 难 就 难 在 脚本 的 调试 和 维护 。 为 了 能 够 更 加 聚焦 于 PHP 脚本 
的 编号， 建议 大 家 还 是 使 用 集成 的 开发 环境 ， 不 仅 上 手 更 快 ， 脚 本 维护 起 来 也 更 方便 。 


2.8.1 IDE 是 什么 
简 而 言 之 ，IDE 为 编码 提供 一 站 式 服务 的 环境 。 一 个 完整 的 IDE 通常 包括 一 个 编辑 器 
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用 于 编辑 和 调试 代码 ， 还 包括 一 个 内 嵌 的 浏览 器 用 于 查看 脚本 的 运行 情况 。 有 的 IDE 还 集 
成 了 版 本 控制 功能 。 
一 般 来 说 ，IDE 具有 如 下 几 个 特点 。 


1. 基于 项 目的 开发 


IDE 的 一 个 关键 特性 是 它 把 一 个 PHP 应 用 程序 看 作 是 一 个 项 目 ， 而 不 仅仅 是 一 组 文 
件 。 在 这 个 概念 下 ，IDE 可 以 帮助 我 们 更 好 地 控制 源码 、 调 试 数 据 库 ， 以 及 规划 某 一 关键 
目录 的 所 在 位 置 。 

2. 调试 


IDE 的 另 一 个 方便 的 特性 是 集成 调试 。 这 样 一 来 ， 可 以 在 编辑 器 中 设置 断 点 ， 当 PHP 
解释 程序 执行 到 这 个 脚本 时 就 会 停止 ， 以 便 检查 局 部 变量 的 值 ， 从 而 诊断 脚本 中 出 现 的 问 
题 。 当然, 也 可 以 在 脚本 中 可 以 使 用 echo 语句 来 检查 值 或 者 也 可 以 使 用 错误 日 志 获 得 变量 
的 值 。 


3. 代码 智能 补 全 


PHP 是 一 种 非常 规则 的 编程 语言 ， 这 意味 着 它 遵循 着 简单 的 模式 。 这 些 模式 不 仅 使 代 
码 易于 编写 ， 也 使 IDE 在 项 目 中 检查 代码 变 得 很 容易 。 此 外 ， 这 一 功能 还 可 以 帮助 我 们 更 
快 更 准确 地 编写 脚本 。 举 个 例子 ， 如 果 我 们 在 项 目 中 定义 了 一 个 名 为 MyClass 的 类 ， 那 么 
当 我 们 在 编辑 器 中 输入 了 关键 词 new 之 后 , 光标 附近 立即 会 出 现 一 个 包括 MyClass 作为 选 
项 的 列表 。 无 论 使 用 何 种 类 型 的 对 象 ，IDE 都 会 立刻 显示 它 的 可 用 方法 和 实例 变量 。 这 应 
该 是 使 用 IDE 而 不 是 文本 编辑 器 的 首要 原因 。 这 种 代码 智能 补 全 功能 可 以 有 效 减少 敲 错 类 
名 、 方 法 名 和 参数 。 

4. 类 视图 

IDE 中 的 代码 智能 引擎 产生 的 另 一 个 作用 是 IDE 可 以 产生 项 目的 类 视图 。 使 用 这 一 功 
能 ， 可 以 方便 地 了 解 当前 项 目 中 已 经 定义 的 所 有 类 ， 而 不 用 去 担心 找 不 到 定义 这 些 类 的 文 
件 。 当 使 用 这 些 类 时 ， 可 以 方便 地 通过 编辑 器 访问 相应 文件 并 显示 相应 类 、 方 法 或 者 实例 
变量 。 

5. 多 语言 支持 

大 多 数 的 IDE 不 仅 支持 PHP 而 且 支 持 如 JavaScript\Structured Query Language( SQL)、 
HyperText Markup Language CHIML ) 和 Cascading Style Sheets (CSS) 等 相关 的 语言 集 。 
因为 HIML 和 CSS 比较 简单 , 所 以 IDE 对 它们 的 支持 是 最 好 的 。 对 于 JavaScript 的 支持 通 
常会 有 兼容 性 问题 也 导致 代码 智能 补 全 失效 ， 但 是 支持 比 不 支持 要 好 。 

6. 源码 控制 


大 多 数 的 IDE 都 支持 一 些 与 源码 控制 系统 的 连接 ， 以 方便 随 着 时 间 的 推移 维护 项 目 文 
件 的 不 同 版 本 ， 并 可 以 标记 某 一 特定 的 版 本 为 发 布 版 本 。 在 团队 环境 中 使 用 源码 控制 系统 
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是 十 分 重要 的 ， 但 这 并 不 是 说 源码 控制 对 于 个 人 使 用 不 重要 。 当 磁盘 瘫痪 或 者 客户 突然 想 
要 以 前 的 版 本 而 不 是 现在 的 版 本 的 时 候 ， 一 个 好 的 源码 控制 系统 就 可 以 发 挥 作用 了 。 

7. FTP/SFTP 集成 

与 源码 控制 相关 的 一 种 功能 是 在 服务 器 中 对 于 最 新 的 代码 使 用 FTP。 这 比 使 用 FTP 客 
户 机 或 者 自己 打包 文件 并 发 送 给 服务 器 然后 再 解 包 要 容易 许多 。 

8， 数据 库 导 航 

有 些 IDE 会 提供 数据 库 导航 功能 。 使 用 这 个 特性 ， 可 以 浏览 应 用 程序 访问 的 数据 库 、 
找到 表格 和 字段 名 并 返回 查询 结果 。 一 些 系统 甚至 可 以 自动 写 入 一 些 数据 库 访问 代码 。 

9. 集成 Web 浏 览 器 

一 些 IDE 支持 集成 Web 浏览 器 ， 可 以 直接 导航 到 正在 使 用 指定 的 附加 参数 编辑 的 
页 面 。 既 可 以 直接 使 用 内 置 在 IDE 中 的 ， 也 可 以 调用 外 部 浏览 器 。 建 议 大 家 还 是 多 使 用 外 
部 浏览 器 ， 因 为 内 置 浏 览 器 对 源码 的 解析 可 能 会 与 用 户 使 用 的 实际 情况 有 差异 ， 从 而 导致 
页 面 解析 出 现 的 效果 并 不 是 想 要 的 样子 。 

10. 片段 


基本 上 所 有 的 IDE 都 提供 了 方便 快捷 地 保存 代码 片段 的 功能 。 所 谓 片段 ， 其 实 就 是 完 
成 小 任务 〈 比 如 在 一 些 输入 中 运行 常规 表达 式 、 连 接 到 数据 库 和 查询 数据 库 ) 的 小 部 分 代 
码 。 读 者 可 以 将 经 常 使 用 的 一 些 代 码 片段 保存 起 来 ， 以 方便 随时 调用 。 


2.8.2 PHP 开发 中 常用 的 IDE 


在 PHP 的 开发 中 ， 比 较 常见 的 IDE 有 如 下 儿 种 。 
1. PHPEclipse 


PHPEclipse 是 基于 Eclipse 独立 开发 的 一 套 IDE。 可 以 方便 地 运行 在 Windows、Linux 
和 Unix 平 台 上 。 支 持 2.8.1 小 节 里 提 到 的 几乎 所 有 的 功能 。 是 一 款 专 注 PHP 项 目 开 发 的 IDE。 


2. Komodo IDE/Edit 


Komodo IDE 是 一 款 商业 软件 ， 除 了 支持 PHP 开发 之 外 ,还 支持 Python、Ruby 和 Perl 
等 语言 的 开发 。 功 能 十 分 的 强大 。 如 果 不 想 付 费 ， 可 以 使 用 Komodo Edit。 这 是 一 款 免 费 
的 脚本 编辑 器 ， 集 成 了 Komodo IDE 的 代码 智能 补 全 功能 。Komodo IDE 令 人 称道 的 一 点 
是 ， 它 集成 了 一 个 正则 表达 式 调 试 器 ， 可 以 方便 地 调试 正则 表达 式 。 

3. Aptana Studio/Plugin 

Aptana Studio 是 一 款 基于 Eclipse 的 开源 的 IDE。 与 Komodo 类 似 的 是 , 它 支 持 多 种 编 
程 语言 ， 可 以 很 方便 地 同时 管理 多 个 不 同 语言 的 项 目 ， 同 时 它 的 代码 智能 补 全 系统 十 分 的 
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健壮 , 对 于 提高 代码 输入 效率 和 代码 准确 性 来 说 , 不 可 多 得 。 另外 , 它 还 为 已 经 熟悉 Eclipse 
开发 流程 的 人 准备 了 Aptana Plugin。 如 果 不 想 让 电脑 上 的 IDE 成 堆 ， 那 么 也 可 以 试 试 它 的 
Plugin， 功 能 与 Studio 版 本 是 一 样 的 。 


4. Zend Studio 


Zend Studio 是 一 款 号 称 专业 的 PHP 项 目 开 发 必 不 可 少 的 IDE。 不 过 , 它 是 一 款 商业 软 
件 ， 使 用 它 可 是 要 付费 的 。 与 PHPEclipse 类 似 的 是 ， 它 也 专注 于 PHP 开发 ， 上 面 提 到 的 
各 种 功能 它 都 支持 。 如 果 是 企业 级 应 用 的 开发 ， 建 议 还 是 用 专业 的 IDE。 不 过 对 于 看 这 本 
书 的 读者 来 说 ， 就 不 是 那么 必要 了 。 


5. PHPEdit 


从 严格 意义 上 来 说 ，PHPEdit 不 能 算是 IDE。 不 过 它 也 有 强大 的 代码 智能 引擎 ， 可 以 
提高 代码 输入 的 效率 和 准确 性 。 因 此 ， 姑 且 把 它 皖 在 这 里 。 

如 果 将 上 面 这 五 种 IDE 按 功能 分 类 的 话 ， 可 以 分 为 两 类 ， 一 类 仅 支 持 PHP 项 目的 开 
发 , 如 Zend Stuido、PHPEclipse 和 PHPEdit; 另 一 类 则 支持 多 种 语言 项 目的 开发 , 如 Komodo 
IDE 和 Aptana Studio。 大 家 可 以 根据 自己 的 需要 取舍 一 下 。 


2.8.3 创建 PHP 项 目 


在 这 一 小 节 里 ， 我 们 选择 Aptana Studio 来 创建 第 一 个 PHP 项 目 。 选 择 Aptana Studio 
的 原因 是 因为 它 强大 的 代码 智能 引擎 ， 可 以 极 大 地 提升 脚本 编写 效率 和 准确 率 。 同 时 ， 其 
项 目 化 的 管理 ， 也 可 以 帮助 大 家 很 快 地 上 手 PHP 编程 。 

现在 Aptana Studio 主推 的 版 本 是 Aptana Studio 3。 大 家 可 以 访问 它 的 官方 网 站 获取 。 
按照 官方 网 站 的 说 法 ，Aptana Studio 3 有 六 个 核心 功能 。 它 们 分 别 是 ， 代码 助 手 、 部 署 向 
导 、 集 成 调试 、 版 本 控制 集成 、 内 置 终端 和 IDE 自 定义 ， 如 图 2-62 所 示 。 
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图 2-62 ”Aptana Studio 3 核心 功能 介绍 
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Aptana Studio 3 在 Windows 操作 系统 中 的 安装 过 程 与 其 他 应 用 程序 没有 太 大 的 区 别 。 
在 安装 过 程 中 ， 建 议 都 使 用 默认 的 配置 ， 直 接 单 击 “ 下 一 步 ” 按 钮 就 好 。 

在 安装 结束 后 ， 第 一 次 打开 Aptana Studio 的 时 候 ， 会 弹出 一 个 对 话 框 让 我 们 指定 默认 
的 工作 区 (workspace) 。 当 指定 了 工作 区 后 ， 就 可 以 看 见 Aptana Studio 的 开始 页 面 了 (如 
图 2-63 所 示 ) 。 在 这 个 页 面 中 ， 可 以 接 入 Aptana 的 讨论 区 、 帮 助 文档 和 Bug 库 。 


翘 aptana 


Aptana Discussions @ Aptana Documentation @ Aptana Bug Database @ 


Fast Reference Notable Recent Issues 
Type Key Summary Greated Revwlution 
国 mn re 201 Fixed 


图 2-63 ”Aptana Studio 开始 页 面 


现在 ， 先 来 认识 一 下 Aptana Studio 的 界面 ， 如 图 2-64 所 示 。 


菜单 栏 工具 栏 


互 或 册 / 莫 准 玖 本 吕 


+ 


缆 训 典 证 卫 奖 互 / 东 址 工 呈 


图 2-64 Aptana Studio 界面 截图 


在 安装 好 Aptana Studio 并 设置 好 默认 工作 区 之 后 , 就 可 以 创建 PHP 项 目 了 。 选择 File 
INew.… | PHP Project 命令 ， 弹 出 如 图 2-65 所 示 的 窗口 。 

在 窗口 出 输入 项 目 名 称 和 使 用 的 PHP 版 本 , 然后 单 击 窗口 下 方 的 Finish 按钮 就 算 成 功 
创建 了 一 个 PHP 项 目 。 在 新 创建 的 PHP 项 目 中 默认 是 没有 任何 文件 的 ， 读 者 可 以 在 PHP 
项 目 创建 PHP、HTML、CSS 和 JS 等 一 切 与 项 目 有 关 的 文件 。 
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Createa new PHP project 


Project name | 
加 Use default location 


on: [C\Users\Tu Wenjia\My Documents\Aptana Studio 3 Workspac 


Compatibility 


prpvaor [EER532 


图 2-65 创建 PHP 项 目 


29 本 题 


(1) 如 果 读 者 使 用 的 是 Microsoft Windows 系列 的 操作 系统 ， 请 使 用 Apache 2.x、PHP 
5.2 和 MySQL 5.x 搭建 PHP 的 运行 环境 。 

(2) 如 果 读 者 使 用 的 是 Microsoft Windows 操作 系统 中 自 带 的 IS 服务 器 ， 请 按照 2.7 
节 中 的 描述 使 用 PHP 5.4 和 MySQL 5.x 来 搭建 PHP 的 运行 环境 。 

(3) 请 按照 2.8 节 中 的 描述 安装 好 集成 开发 环境 。 


。42 。 


第 3 章 ”动手 写 第 一 个 PHP 脚本 


所 谓 PHP 脚本 ， 其 实 就 是 一 串 指令 ， 告 诉 PHP 处 理 引 擎 应 该 完成 什么 动作 。 理 论 上 
来 说 ，PHP 脚本 可 以 只 包含 一 条 命令 ， 也 可 以 包含 成 千 上 万 条 命令 ， 这 完全 取决 于 读者 的 
需要 。 对 于 一 个 PHP 脚本 来 说 ,PHP 处 理 引擎 是 按照 从 上 到 下 、 从 左 到 右 的 顺序 一 条 一 条 
处 理 的 ， 直 到 引擎 指针 指向 脚本 的 最 后 一 行 命令 。 

那么 ， 我 们 可 以 用 PHP 脚本 做 些 什么 事情 呢 ? 按照 本 书 1.2 节 中 的 说 法 ， 我 们 可 以 编 
写 PHP 脚本 实现 在 特定 的 网 页 显示 特定 的 内 容 、 将 用 户 在 表单 中 填写 的 内 容 存 入 数据 库 、 
将 某 目 录 中 的 文件 备份 到 指定 的 存储 设备 上 。PHP 几乎 无 所 不 能 ， 只 要 读者 肯 下 功夫 ， 学 
习 PHP 是 一 件 十 分 简单 并 且 充 满 乐趣 的 事 。 

在 本 章 里 ， 我 们 将 要 动手 写 下 第 一 个 PHP 脚本 。 


3.1 何谓 PHP 命令 


按照 表现 形式 的 不 同 ，PHP 命令 可 以 分 为 简单 命令 和 复杂 命令 西 种。 如 何 判断 一 条 命 
令 是 简单 还 是 复杂 呢 ? 


3.1.1 简单 命令 


每 条 简单 的 PHP 命令 都 在 告诉 PHP 处 理 引擎 执行 一 个 动作 。 最 常见 的 PHP 命令 就 是 
echo 命令 ， 它 的 功能 是 显示 和 输出 信息 。 在 第 1 章 里 ， 我 们 就 已 经 见 过 这 条 命令 了 。 

现在 再 来 详细 地 看 一 下 这 个 命令 。 

【 例 3.1】 echo 命令 。 


eobho "Hi"s 


在 这 条 命令 中 ， 有 三 个 部 分 组 成 。 它 们 分 别 是 命令 关键 字 echo、 命 令 对 象 下 和 行 结 
束 符 〈;) 。 当 PHP 处 理 引 擎 读 到 这 条 命令 时 ， 它 首先 会 看 到 这 条 命令 的 关键 字 ， 通 过 关 
键 字 了 解 命令 要 求 完成 的 动作 ， 然 后 再 读 取 命令 的 对 象 ， 并 按照 命令 关键 字 的 要 求 完成 对 
对 象 的 处 理 ， 最 后 引擎 会 读 取 行 结束 符 来 结束 对 这 条 命令 的 执行 。 

刚才 说 到 ,echo 命令 的 功能 是 显示 和 输出 信息 ,那么 当 PHP 处 理 引擎 读 到 这 条 命令 时 ， 
就 会 输出 一 个 简单 的 字符 串 “Hi”。 

这 个 例子 十 分 好 懂 ， 也 没有 什么 好 讲 的 。 不 过 有 的 同学 可 能 对 行 结束 符 产生 了 兴趣 。 
行 结束 符 和 平常 Word 文档 里 的 回 车 符 有 什么 本 质 上 的 区 别 么 ? 为 了 讲 清楚 这 个 问题 ， 青 
来 看 几 条 命令 。 
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【 例 3.2】 行 结 束 符 。 
主 echo "Great'! 
四 I hope I can finally get there!"; 
7 echo "Great!"; echo "Well done!™"; 

看 到 这 里 ， 有 的 同学 可 能 就 不 淡定 了 : 这 到 底 算 是 三 条 命令 还 是 两 条 命令 呢 ? 嗯 ， 这 
个 问题 问 得 好 。 其 实 答案 也 很 简单 ， 那 就 是 只 有 当 行 结束 符 出 现 的 时 候 ， 一 条 命令 才 算 结 
束 , 无 论 一 条 命令 被 切 成 了 儿 段 写 在 了 儿 行 里 。 同 理 , 若干 条 简单 命令 只 有 行 结束 符 齐全 ， 
也 可 以 挤 在 一 行 里 抱团 取暖 ， 就 像 例 3.2 中 的 第 4 行 一 样 。 即 便 如 此 ， 还 是 建议 大 家 一 行 
只 写 一 条 简单 命令 ， 这 样 在 后 期 进行 错误 定位 的 时 候 会 好 过 得 多 。 

PHP 引擎 其 实 无 从 知晓 代码 的 内 容 ， 它 只 知道 寻找 行 结束 符 。 在 两 个 行 结束 符 之 间 的 
内 容 就 会 被 PHP 引擎 当成 一 条 命令 加 以 执行 。 于 是 上 面 这 条 命令 的 结果 就 是 另 一 个 简单 的 
字符 串 “Great!IhopeIcan finally get there!”。 

好 奇 的 同学 可 能 又 要 问 了 : 如 果 在 一 个 脚本 里 一 个 行 结束 符 都 没有 的 话 ， 是 不 是 所 有 
的 代码 就 会 一 起 执行 呢 ? 答案 是 肯定 的 ， 但 是 你 却 看 不 到 你 想 要 的 结果 ， 看 到 的 只 是 如 下 
的 一 条 报错 信息 : 


Parse error: expecting "," or ";" in file.php on line 6 


在 报错 信息 中 ， 你 会 看 到 出 错 的 文件 名 和 具体 的 行 号 ， 以 及 可 能 解决 问题 的 办 法 。 通 
常情 况 下 ， 在 每 一 条 命令 结尾 处 加 上 一 个 分 号 就 能 解决 这 个 问题 。 

对 于 一 个 只 有 几 条 命令 组 成 的 PHP 脚本 文件 来 说 ， 定 位 错误 是 十 分 容易 的 一 件 事 情 。 
但 是 通常 情况 下 , 一 个 PHP 脚本 怎么 着 也 得 有 个 上 百 条 命令 。 因 此 选用 一 款 可 以 显示 行 号 
的 编辑 器 就 成 了 一 个 明智 的 选择 。 否 则 ， 你 就 只 能 从 上 往 下 一 条 一 条 地 数 了 。 


3.1.2 复杂 命令 


把 若干 条 简单 命令 放 到 一 对 花 括 号 里 ， 这 些 命令 就 组 成 了 一 个 复杂 命令 。 一 个 复杂 命 
令 通常 包含 若干 条 简单 命令 ， 甚 至 还 会 嵌 套 一 些 复杂 命令 。 最 常见 的 复杂 命令 块 就 是 条 件 
命令 ， 只 有 满足 特定 条 件 时 ， 花 括号 中 的 简单 命令 才 会 被 执行 ， 如 例 3.3 所 示 。 

【 例 3.3】 下 条 件 命令 。 


if (time is in the morning) 


get up; 

brush my teeth; 
wash my face; 
put on my jacket; 
go to work; 


|: 


在 上 面 这 个 例子 里 只 出 现 了 一 对 花 括 号 。 这 对 花 括号 中 包含 了 5 条 简单 命令 。 这 个 例 
子 可 以 做 如 下 的 解读 : 


早晨 ， 我 起 床 后 会 先 刷 牙 ， 再 洗脸 ， 然 后 穿 上 我 的 夹克 出 门 上 班 。 
在 这 句 简单 的 描述 中 ， 我 们 可 以 发 现 作为 条 件 的 时 间 是 早晨 。 也 就 是 说 只 有 早晨 ， 我 
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才 会 做 如 下 的 动作 : 起床、 刷牙 、 洗 脸 、 穿 衣 和 出 门 上 班 。 于 是 ， 需 要 把 条 件 写 在 花 括 号 
外 的 让 子 句 中 ,然后 把 当 条 件 满足 时 需要 完成 的 动作 依次 写 在 花 括号 内 ， 从 而 完成 一 个 复 
杂 的 条 件 命 令 。 

对 于 一 条 复杂 命令 来 说 , PHP 会 一 次 性 读 取 这 条 复杂 命令 所 有 的 内 容 。 值 得 注意 的 是 ， 
花 括 号 后 面 是 不 需要 加 行 结束 符 〈;) 的 。 

另外 ， 大 家 还 要 注意 务必 让 花 括 号 内 的 每 条 语句 都 缩 进 若干 字符 并 使 它们 保持 左 对 
齐 。 这 个 要 求 并 不 是 必须 的 。 但 是 ， 如 果 你 和 其 他 的 同事 都 在 编辑 同一 个 PHP 脚本 ， 为 了 
他 人 阅读 的 方便 还 是 建议 大 家 照 做 。 


3.2 ”如 何 写 代码 


第 1 章 里 提 到 动态 网 页 这 个 概念 。 为 了 使 网 页 “ 动 ” 起 来 ， 就 得 在 HTML 代码 中 插入 
PHP 脚本 ， 然 后 将 这 些 内 内 PHP 脚本 的 HTML 文件 保存 为 扩展 名 为 .php 的 文件 。 如 此 一 
来 ，PHP 处 理 引擎 才 会 处 理 文件 中 的 PHP 脚本 。 

本 节 将 讨论 一 个 PHP 脚本 应 该 包括 的 元 素 。 


3.2.1 PHP 标记 对 


只 有 当 PHP 脚本 被 插入 扩展 名 为 .php 的 HTML 文件 中 时 ，PHP 引擎 才 会 处 理 这 些 脚 
本 。 那 么 应 该 把 这 些 脚 本 插入 到 HTML 文件 中 的 什么 地 方 呢 ? 先 来 看 例 3.4 中 的 代码 。 
【 例 3.4】 PHP 标记 对 。 


1 <?php 

过 梧 二 

3 PHP statements 
3 

SU 


所 有 的 PHP 脚本 都 应 该 被 包含 在 如 例 3.4 所 示 的 标记 对 中 。 读 者 也 可 以 使 用 “<?” 和 
“?>” 来 标记 一 个 PHP 脚本 的 起 止 。 前 提 是 修改 了 php.ini 文件 中 关于 启用 短 标记 对 的 相关 
内 容 。 

一 般 来 说 ， 使 用 短 标记 并 不 是 一 个 好 主意 。 如 果 把 使 用 短 标 记 对 的 HIML 文件 转移 到 
一 台 没 有 启用 短 标记 对 的 服务 器 上 , 那么 所 有 的 PHP 脚本 都 会 失效 。 尤 其 是 对 于 租用 服务 
器 的 开发 者 来 说 ， 这 样 做 的 后 果 是 致命 的 ， 因 为 大 多 数 供应 商 并 不 允许 修改 php.ini 文件 。 
这 样 一 来 ,编码 时 少 敲 几 个 字母 的 代价 也 忒 大 了 些 。 所 以 还 是 建议 大 家 尽量 使 用 完整 的 PHP 
标记 对 。 

第 1 章 的 结尾 曾经 提 到 : 如 果 用 户 通过 浏览 器 发 出 访问 请 求 , PHP 标记 对 间 的 所 有 PHP 
脚本 都 会 被 送 到 PHP 处 理 引擎 进行 处 理 。 然 后 服务 器 将 经 过 处 理 的 页 面 下 发 到 发 出 请 求 的 
浏览 器 。 该 页 面 中 所 有 的 PHP 脚本 都 已 经 被 替换 成 了 相应 的 处 理 结果 。 在 浏览 器 里 通过 查 
看 源 代码 的 方式 是 无 法 看 到 任何 PHP 脚本 的 。 

例如 ， 读 者 可 以 在 HIML 代码 中 加 入 如 例 3.5 所 示 的 PHP 脚本 ， 然 后 将 HTML 代码 
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保存 为 一 个 PHP 文件 。 
【 例 3.5】 PHP 脚本 。 


未 <?php 
所 echo "This line is brought to you by PHP."; 
3 2 


当 用 户 请 求 该 页 面 时 ,服务 器 会 先 查 看 文件 的 扩展 名 。 当 服务 器 发 现 该 文件 是 一 个 PHP 
文件 时 ， 安 装 在 服务 器 上 的 PHP 处 理 引擎 就 会 检查 该 文件 里 的 PHP 标记 对 、 执 行 标记 对 
中 的 脚本 、 并 输出 相应 的 结果 。 在 本 例 中 ， 服 务 器 上 的 PHP 处 理 引 擎 会 执行 PHP 标记 对 
中 的 echo 命令 ， 并 输出 处 理 结果 ， 也 就 是 “This line is brought to you by PHP” 这 句 话 。 

当 执 行 完 文 件 中 应 该 执行 的 所 有 脚本 后 ， 服 务 器 会 用 脚本 的 执行 结果 替换 相应 的 脚 
本 ， 然 后 将 处 理 后 的 HIML 文件 下 发 到 用 户 的 浏览 器 中 。 用 户 就 能 看 到 上 面 那 句 话 了 。 


3.2.2 ”注释 脚本 


看 到 这 一 节 的 标题 ， 有 的 同学 会 问 : 为 什么 要 注释 脚本 呢 ? 

注释 对 于 脚本 来 说 十 分 重要 。 通 常情 况 下 ， 我 们 会 使 用 注释 来 描述 代码 ， 告 诉 阅读 脚 
本 的 人 某 一 段 代码 可 以 实现 的 功能 以 及 该 功能 是 如 何 实现 的 。 当 脚本 十 分 复杂 ， 让 人 无 法 
很 快 读 懂 时 ， 注 释 就 显得 尤为 重要 了 。 但 是 如 果 代 码 只 有 自己 一 个 人 在 维护 ， 那 么 是 不 是 
就 不 用 注释 了 ? 自己 写 的 代码 难道 自己 还 看 不 懂 吗 ? 对 于 这 个 问题 ， 我 只 能 用 一 句 俗语 来 
回答 : “好 记性 不 如 烂 笔头。” 既 然 我 们 可 以 很 方便 地 在 脚本 旁边 注 明 一 下 某 段 脚本 的 功 
能 ， 为 什么 不 呢 。 更 何况 ， 脚 本 会 变 得 越 来 越 复 杂 ， 总 有 一 天 会 需要 很 多 的 人 来 一 起 维护 。 
写 上 注释 就 可 以 避免 出 现代 码 无 法 维护 的 情况 ， 提 高 代码 的 利用 效率 。 

所 谓 注 释 ， 其 实 就 是 写 在 脚本 旁边 用 于 说 明代 码 的 一 段 文字 。PHP 处 理 引擎 在 箭 到 注 
释 时 会 直接 忽略 。 也 就 是 说 ， 注 释 一 定 是 给 人 看 的 ， 那 么 写 注释 的 时 候 言 简 意 凡 就 显得 十 
分 必要 了 。 那 么 PHP 处 理 引 擎 如 何 区 别 脚本 和 注释 呢 ? 还 是 通过 一 个 例子 来 说 明 一 下 。 

【 例 3.6】 注释 示例 。 

1 ， /* ”在 这 儿 写 注释 

2 ”在 这 儿 写 更 多 的 注释 */ 

在 例 3.6 中 ， 我 们 看 到 了 如 下 的 两 个 符号 : “/*#*” 和 “*/”， 这 样 的 注释 标记 称 为 长 注 
释 标 记 。PHP 处 理 引 擎 在 看 到 这 一 对 符号 时 ， 就 会 直接 忽略 它们 之 间 的 所 有 内 容 。 大 家 可 
以 在 开始 写 脚本 之 前 ， 在 开头 的 地 方 注释 一 段 ， 写 一 写 脚本 的 名 字 、 描 述 、 作 者 信息 和 写 
作 时 间 等 信息 ， 以 后 查找 起 来 也 会 非常 方便 。 例 3.7 就 是 一 段 脚 本 说 明 。 

【 例 3.7】 脚本 说 明 。 


1 /* name: hello.php 

description: Displays "Hello World!" on a web page. 
written by: Joe Programmer 

created on: Feb 15t, 2012 

modified on: Mar 15™, 2012 


an 心 wm 


El 
值得 注意 的 是 ， 长 注释 标记 不 支持 嵌 套 。 也 就 是 说 ， 如 果 出 现 了 如 例 3.8 这 样 的 注释 
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标记 ，PHP 会 报错 。 

【 例 3.8】 错误 的 注释 嵌 套 。 

1 ”/* 这 是 一 条 注释 

瑟 /* 这 是 另 一 条 注释 */ 

37°= 

有 人 可 以 会 间 了 ， 这 不 是 挺 工整 的 吗 ， 为 什么 会 报错 呢 ? 我 们 来 分 析 一 下 : 按照 之 前 
的 说 法 , PHP 处 理 引 擎 在 见 到 “/#*” 符 号 时 ， 就 会 忽略 之 后 的 所 有 内 容 , 直到 它 遇 到 了 “#/” 
符号 。 这 样 看 来 ， 在 例 3.8 中 ，PHP 处 理 引 擎 会 把 第 一 行 的 “#” 和 第 二 行 的 “*/” 当 成 是 
注释 的 开始 和 结尾 ， 而 把 第 二 行 开头 的 “/*” 当 成 了 注释 的 一 部 分 。 那 么 第 三 行 的 “*/” 
就 形 只 影 单 无 人 顾 了 。 了 PHP 处 理 引 擎 也 会 因为 无 法 处 理 这 个 形 只 影 单 的 标识 符 而 报错 。 

对 于 注释 内 容 如 例 3.7 这 样 比较 多 的 情况 来 说 ， 这 个 注释 标记 还 显得 不 是 很 累 效 。 如 
果 注 释 只 有 一 行 ， 还 要 陪 上 4 个 字符 的 注释 标记 对 ， 效 率 实在 是 太 低 了 。 其 实 PHP 还 提供 
了 两 种 短 注释 ， 标 识 符 是 井 号 (#) 或 双 斜 杜 (//) ， 如 例 3.9 所 示 。 

【 例 3.9】 短 注释 。 


1 # This is a comment. 
2 //This is another comment. 


那么 这 两 种 标识 符 有 什么 区 别 呢 ? 井 号 (#) 只 能 用 在 一 行 的 开始 ， 而 双 斜 枉 〈/) 可 
以 用 在 一 行 的 中 间 。 当 需要 在 某 一 行 命令 后 进行 注释 时 ， 可 以 使 用 双 斜 杠 (/) ， 如 例 3.10 
所 示 。 

【 例 3.10】 双 斜 杠 注释 符 可 以 用 在 一 行 的 末尾 。 


十 <?php 
2 echo "Hello World!'"; // 打 印 “Hello World!” 
3 Es 


在 本 书 中 ， 这 三 种 注释 标识 都 会 用 到 。 为 了 统一 风格 ， 也 为 了 帮助 大 家 养 成 注释 脚本 
的 习惯 。 本 书 做 出 如 下 的 规定 : 
口 脚本 的 开始 使 用 长 注释 书写 脚本 说 明 。 
口 若 一 个 脚本 中 包含 有 若干 个 模块 ， 在 每 个 模块 开始 前 用 短 注 释 标识 〈#) 说 明 模块 
的 功能 
口 在 重要 的 命令 行 后 用 短 注释 标识 (1) 说 明 该 命令 行 的 作用 。 


3.3 ”实战 练习 : 向 世界 说 Hellol 


虽然 之 前 就 已 经 提 到 过 echo 网 和 E 和 用 法 , 但 是 并 没有 形成 一 个 系统 的 概念 。 为 
了 更 好 地 使 用 这 条 十 分 常用 的 命令 , 很 有 必要 在 正式 地 用 PHP 网 本 向 世界 说 你 好 之 前 系统 
地 讲解 一 下 这 条 命令 。 


3.3.1 ”echo 命令 初 识 


这 条 命令 可 以 说 是 编写 PHP 脚本 必用 的 命令 之 一 。 没有 哪个 脚本 在 被 执行 之 后 不 输出 
信息 的 。 一 旦 需要 输出 信息 ， 就 一 定 会 用 到 这 条 命令 。 例 如 ， 编 写 了 一 段 在 操作 系统 里 查 
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找 某 个 特定 文件 的 脚本 。 查 找 的 过 程 当然 是 不 可 见 的 ， 但 是 如 果 连 查找 的 结果 也 不 输出 的 
话 ， 如 何 知道 这 个 脚本 是 不 是 起 作用 了 ， 要 查找 的 文件 是 不 是 已 经 找到 了 呢 ? 对 于 这 样 的 
脚本 ， 通 常 需要 输出 的 信息 包括 ， 查 找 结果 和 需要 查找 文件 的 文件 名 等 。 如 果 在 操作 系统 
中 查找 到 了 该 文件 、 则 还 需要 输出 该 文件 的 存储 路 径 、 存 储 时 间 、 最 后 修改 时 间 和 摘要 信 
息 。 这 样 一 来 ， 我 们 才能 知道 编写 的 脚本 是 不 是 起 了 作用 、 是 否 找到 了 需要 的 文件 以 及 文 
件 的 基本 情况 。 

所 以 说 ，echo 命令 是 十 分 重要 也 是 十 分 必要 的 一 条 命令 。 按 照 之 前 的 示例 中 书写 的 样 
我 们 可 以 总 结 出 echo 命令 的 基本 样式 。 

【 例 3.11】 echo 命令 的 基本 样式 。 


echo outputl, output2, output3, ... 


在 使 用 echo 命令 时 ， 一 定 要 注意 以 下 几 点 : 

口 输出 的 对 象 ， 也 就 是 例 3.11 中 的 output 参数 一 定 是 一 个 数字 或 者 字符 串 。 所 谓 数 
字 就 是 像 1 或 者 234 这 样 的 数字 (如 echo 93;) ， 而 字符 串 则 是 一 串 包含 在 引号 内 
的 字符 (如 echo "Hello World!";) 。 关 于 引号 的 使 用 ,将 在 第 7 章 讲 字符 串 型 变量 
的 时 候 会 进行 详细 地 讲解 。 

口 理论 上 来 讲 ， 一 条 echo 命令 可 以 输出 的 对 象 是 无 限 的 ， 但 是 当 输 出 对 象 多 于 两 个 
时 ， 需 要 在 相 邻 的 输出 对 象 之 间 加 上 一 个 逗号 。 千 万 注意 ， 不 要 画 蛇 添 足 的 在 远 
号 后 面 加 个 空格 ， 和 否则 脚本 会 报错 。 

口 空格 也 是 一 种 字符 ， 可 以 通过 在 空格 前 后 添加 引号 来 输出 (如 echo"” ";)。 

表 3-1 所 示 列 出 了 使 用 一 些 echo 命令 和 它们 的 输出 结果 。 


表 3-1 echo 命 令 的 输出 结果 


式 


echo 命令 输出 结果 
echo 159; 159 
echo "Hello World!": Hello World! 
echo "Hello","World!": HelloWorld! 
echo "Hello"," ","World!"; Hello World! 


在 使 用 echo 命令 时 ， 还 要 注意 一 些 诸如 “nn”、“\t” 和 “\” 这 样 的 特殊 字符 和 转 义 
字符 。 具 体 的 内 容 会 在 第 7 章 讲解 字符 串 类 型 的 变量 时 进行 详细 地 讲解 。 

到 这 里 , 使 用 echo 命令 需要 注意 的 相关 内 容 都 呈现 给 大 家 了 。 是 不 是 很 简单 呢 ? 有 没 
有 学 会 靠 事 实 来 说 话 。 下 面 就 让 我 们 进入 到 实战 练习 吧 。 


3.3.2 ”实战 练习 


向 世界 说 Hellol 


在 本 小 节 里 , 我 们 将 使 用 PHP 基础 知识 来 写 一 段 脚本 ， 用 不 同 的 语言 向 世界 说 Hellol 

在 示例 脚本 中 ， 会 出 现 一 些 之 前 没有 见 过 的 内 容 ， 比 如 说 数组 、 再 比如 说 循环 等 等 。 
这 些 都 是 PHP 的 基础 知识 ， 大 家 会 在 后 续 的 章节 中 学 到 。 另 外 ， 脚 本 中 涉及 到 关于 HTML 
和 JavaScript 的 相关 知识 ， 书 中 会 附带 着 给 予 相应 的 解释 ， 应 该 不 会 给 大 家 的 阅读 带 来 影 
响 。 不 过 ， 这 里 也 有 一 个 小 建议 ， 那 就 是 尽 可 能 多 的 掌握 HTML 和 JavaScript， 特 别 是 在 
HTML 5 大 行 其 道 的 年 代 ， 这 一 点 非常 的 重要 。 
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好 了 ， 言 归 正 传 。 首 先 ， 先 来 新 建 一 个 名 为 01_hello_world.php 的 文件 ， 然 后 在 文件 
中 写 下 一 些 HTML 代码 : 


<!DOCTYPE html> 


<html> 
<head> 
<meta charset="UTF-8"> 
<title>Hello World!</title> 
<style> 
#msg { 
font-size:500%; 
margin-left:60px; 
width:900px; 
height:300px; 
line-height:300px; 
text-align:center; 
overflow:hidden; 
} 
#lang { 
font-size:150%; 
margin-left:120px; 
height:80px; 
line-height:80px; 
text-align:right; 
} 
</style> 
</head> 
<body> 
<div id="msg"></div> // 输 出 问候 的 位 置 
<div id="lang"></div> // 输 出 该 问候 对 应 语言 的 位 置 
</body> // 在 该 标签 前 插入 PHP 脚本 
</html> 


在 上 面 这 份 HIML 文件 中 ,定义 了 两 个 div 标签 ， 分 别 用 来 存放 问候 语 和 该 问候 语 对 
应 的 语言 。 
接 下 来 ， 需 要 在 “</body>” 标 签 前 插入 如 下 的 PHP 脚本 : 
<?php 
$msgArray = array('Hello World!', 
"世界 ， 您 好 ! '， 
"世界 ， 您 好 ! '， 
"三 克 世 为 这， 世界 !，， 
"处 司 植 台 8.5， 
"Bonjour le monde!', 
"Hallo Welt!', 
‘gz0900 ion, 
"saluton mondo'); // 定 义 了 一 个 数组 ， 用 于 存放 问候 语 


$langArray = array('English', 
"简体 中 文 "， 
“繁体 中 文史 
"日 本 语 "， 
, 嫩 子 oj， 
"franSaise'v 
" Deutsch '" ， 
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"39393935D ' 


"Esperanto ') ; // 定 义 了 另 一 个 数组 ， 用 于 存放 问候 语 对 应 的 语言 
$idPrev = 0; 


/* 因 为 在 下 面 的 循环 里 使 用 了 rand () 产生 的 随机 数 ， 为 了 避免 重复 出 现 两 句 相同 的 问候 
这 里 定义 了 一 个 比较 变量 */ 


for ($i=0;$i<=10;$i++){ / /开启 一 个 循环 ， 用 于 随机 输出 问候 语 
$id = rand(0,8); // 获 取 一 个 8 以 下 的 随机 数 
dEl(SIdPrev Sra) // 判 断 当 前 循环 中 产生 的 随机 数 与 上 一 次 循环 中 产 
生 的 循环 次 数 是 否 相同 
$id = $idPrev + 1; 
if($id > 8) 
{ 
gid = 
} 
lL //<= 中 产生 的 随机 数 是 否 相 同 


Smsg = $msgArray[$id]; // 获 取 产 生 的 随机 数 对 应 的 问候 语 
$lang = $langArray[$id]; // 获 取 产 生 的 随机 数 对 应 的 问候 语 使 用 的 语言 
$idPrev = $id; // 为 下 一 次 循环 比较 随机 数 做 准备 
echo '<script 
language="javascript">document .getElementById("msg") .innerHTML="' .$msg. 
ACEIBE 7 
echo '<script 
language="javascript">document .getElementById("lang") .innerHTML=""' .$lan 
a </script> 


/* 上 面 使 用 学 习 到 的 echo 命令 输出 了 两 个 字符 串 。 在 这 两 个 字符 串 中 我 们 使 用 了 
JavaScript 脚本 。 这 个 脚本 的 向 HTML 文件 中 的 ID 为 msg 的 DIV 标签 输出 
变量 $msg 的 内 容 ， 同 时 向 ID 为 lang 的 DIV 标签 输出 变量 $lang 的 内 容 */ 


echo str_repeat(' ',1024*64); “ // 重 复 输 出 空格 以 填 满 缓存 
flush() // 刷 新 缓存 
sleep (1) // 停 顿 一 秒 钟 


上 
2> 


当 运 行 上 面 这 段 脚本 后 ， 会 发 现 浏览 器 中 出 现 了 在 数组 中 定义 的 问候 语 及 其 使 用 的 语 
如 图 3-1 所 示 。 


图 3-1 Hello World (繁体 中 文 ) 
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这 是 怎么 做 到 的 呢 ? 一 起 来 看 看 代码 : 

在 上 面 这 段 脚本 中 ， 首 先 定义 了 两 个 数组 ， 一 个 用 于 存放 问候 语 ($msgArray) ， 另 
一 个 则 用 于 存放 语言 ($langArray) 。 注 意 ， 这 两 个 数组 中 的 问候 语 和 语言 是 一 一 对 应 的 ， 
以 便于 后 续 从 数组 中 取 值 时 得 到 的 结果 也 是 一 一 对 应 的 。 

由 于 这 些 问 候 语 不 是 按照 它们 在 数组 中 的 顺序 在 浏览 器 中 出 现 的 。 为 了 防止 同一 问候 
语 连 续 多 次 出 现 的 情况 ， 我 们 定义 了 一 个 变量 ， 用 于 存放 当前 问候 语 对 应 的 数组 索引 。 该 
变量 会 在 随机 产生 下 一 名 问候 语 的 数组 索引 时 使 用 。 即 $idPrev。 这 个 变量 用 来 存储 上 一 次 
显示 的 问候 语 在 数组 中 的 索引 。 如 果 当 前 问候 语 在 数组 中 的 索引 与 该 变量 的 值 相同 ， 则 该 
变量 的 值 加 1， 并 将 加 1 后 的 结果 赋值 给 存储 当前 索引 的 变量 。 

接 下 来 , 我 们 定义 了 一 个 for 循环 用 来 输出 问候 语 到 浏览 器 。 这 个 for 循环 由 四 步 构成 ， 
分 别 是 : 

(1) 随机 生成 索引 并 比较 与 上 一 次 显示 的 问候 语 在 数组 中 的 索引 。 

(2) 根据 生成 的 索引 在 两 个 数组 中 查找 相应 的 问候 语 及 其 使 用 的 语言 。 

(3) 将 查找 到 的 问候 语 及 其 使 用 的 语言 输出 到 浏览 器 。 

(4) 让 系统 在 刷新 缓存 后 停顿 1 秒 钟 ， 再 次 执行 for 循环 。 

第 一 步 ， 定 义 一 个 变量 8id， 用 于 存储 使 用 rand0 函 数 生 成 的 一 个 8 以 内 的 随机 数 。 若 
变量 $id 的 值 与 $idPrev 相同 ， 则 将 变量 $idPrev 的 值 加 1 并 将 计算 结果 赋值 给 变量 $id， 若 
此 时 变量 $id 的 值 又 大 于 8， 则 将 变量 $id 的 值 定 为 8。 这 样 一 来 ， 就 不 会 出 现 查 找 不 到 问 
候 语 的 情况 了 。 

第 二 步 ， 根 据 生 成 的 索引 在 两 个 数组 中 查找 相应 的 问候 语 及 其 使 用 的 语言 。 具 体 的 做 
法 是 ， 先 定义 两 个 变量 Smsg 和 $lang， 然 后 使 用 在 第 一 步 中 生成 的 变量 $id 的 值 做 为 索引 ， 
在 数组 SmsgArray 和 $langArray 中 查找 相应 的 问候 语 和 使 用 的 语言 ， 并 将 查找 到 的 内 容 赋 
值 给 变量 $msg 和 $lang。 

第 三 步 ， 使 用 echo 语句 输出 两 条 JavaScript 语句 ， 用 来 向 HTML 代码 中 的 两 个 DIV 
标签 添加 相应 的 内 容 。 这 时 ， 浏 览 器 中 就 出 现 了 一 条 问候 语 及 该 问候 语 使 用 的 语言 ， 如 图 
3-1 所 示 。 

第 四 步 ， 使 用 fushO0 和 sleepO 两 个 函数 用 来 刷新 系统 缓存 和 让 服务 器 暂时 休息 。 关 于 
这 两 个 函数 的 具体 内 容 ， 大 家 可 以 查看 PHP 官网 上 的 介绍 。 

至 此 ， 一 次 循环 就 完成 了 。 浏 览 器 中 的 信息 会 每 隔 一 秒 钟 刷新 一 次 。 在 刷新 10 次 后 ， 
浏览 器 中 的 信息 就 不 再 变化 了 。 读 者 可 以 尝试 着 向 数组 中 添加 更 多 的 内 容 ， 同 时 修改 循环 
语句 中 的 变量 $i 的 最 大 值 ， 从 而 展现 更 加 丰富 的 信息 。 

在 这 段 脚本 中 ， 有 很 多 的 概念 ， 比 如 数组 、 循 环 和 条 件 判 断 等 ， 都 是 大 家 第 一 次 遇 到 。 
如 果 不 懂 ， 也 没有 关系 ， 后 续 的 章节 中 对 这 些 内 容 进 行 详细 地 讲解 。 现 在 大 家 只 用 看 看 热 
阅 就 好 。 


3.4 习 题 


(1) 请 使 用 echo 命令 在 浏览 器 中 输入 下 面 这 一 段 文字 : 
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"PHP 是 一 个 免费 开源 的 项 目 ,无 论 是 在 Web 服务 器 上 部 署 PHP 引擎 , 还 是 使 用 PHP 代码 编写 网 站 ， 
你 都 不 需要 花费 一 分 钱 。 天 下 到 底 还 是 有 免费 的 午餐 。" 


(2) 请 在 刚才 编写 的 脚本 中 使 用 "/" 添 加 一 段 注释 ， 注 释 内 容 如 下 : 
"这 是 一 条 注释 。" 

(3) 请 将 如 下 的 内 容 以 注释 的 形式 添加 到 脚本 中 : 

项 目 名 称 : 向 世界 说 你 好 

负责 人 : 张 小 二 


开始 时 间 : 2013 年 11 月 24 日 星期 日 
结束 时 间 : 2013 年 11 月 25 日 星期 一 


Ry 
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看 到 这 个 标题 ， 也 许 很 多 同学 都 会 好 奇 ， 难道 要 学 代数 吗 ? 怎么 又 是 变量 ， 又 是 常量 
的 呢 ? 其 实 所谓 的 编程 说 白 了 就 是 代数 加 上 一 些 条 件 判断 和 复 用 ， 它 的 核心 就 是 代数 。 而 
常量 和 变量 就 是 代数 这 棵 树 上 结 出 的 两 洒 并 蒂 双 生花 。 

那么 在 进入 这 一 章 的 主要 内 容 之 前 ， 先 来 复习 一 下 代数 里 关于 常量 和 变量 的 定义 吧 。 
维基 百科 把 常量 定义 为 不 变 的 量 ， 其 值 是 恒定 不 变 的 。 与 之 相对 的 一 个 概念 就 是 变量 ， 其 
值 会 随 着 条 件 的 改变 而 发 生变 化 。 在 数学 中 ， 比 较 著名 的 常量 包括 : 圆周 率 (x) 、 欧 拉 
数 (e) 和 黄金 分 割 率 (p) 。 

掌握 了 这 些 基础 知识 之 后 , 我 们 就 明确 了 本 章 的 主题 。 基 于 数学 中 常量 与 变量 的 定义 
大 家 大 概 了 解 在 PHP 中 的 常量 和 变量 所 指 为 何 。 所 谓 常 量 应 该 也 是 恒定 不 变 的 量 , 而 所 谓 
变量 则 是 随 条 件 变 化 而 发 生变 化 的 量 。 基 于 这 样 的 认识 ， 随 后 的 两 节 内 容 将 深入 地 介绍 在 
PHP 语 境 下 的 常量 与 变量 到 底 是 什么 。 


4.1 什么 是 常量 


在 PHP 中 , 常量 通常 是 一 个 包含 固定 值 的 量 , 包含 在 常量 中 的 值 不 会 随 脚本 中 其 他 因 
素 的 改变 而 改变 。 当 读者 设 定 了 一 个 常量 的 值 之 后 ， 在 脚本 的 任何 地 方 引用 这 个 值 ， 其 结 
果 都 是 一 致 的 。 例 如 ， 在 脚本 中 定义 了 一 个 常量 Weather， 并 将 其 值 设 定 为 Sunny。 那 么 在 
脚本 中 的 任何 地 方 引 用 Weather， 系 统 都 会 告诉 我 们 天 气 晴朗 。 其 实 ，Weather 更 多 的 时 候 
是 当做 变量 使 用 的 ， 因 为 天 气 是 会 变化 的 。 这 也 是 为 什么 中 文 里 有 那么 多 描绘 气候 变幻 的 
成 语 了 。 


4.1.1 如 何 定义 常量 


在 PHP 脚本 中 ， 可 以 根据 需要 自行 定义 常量 。 例 4.1 所 示 展 示 了 如 何 创 建 一 个 常量 。 
【 例 4.1】 定义 常量 。 


过 define ("constantname", "constantvalue"); 


例 4.1 所 示 的 意思 是 用 define 方法 创建 一 个 名 为 constantname， 值 为 constantvalue 的 
常量 。 在 这 里 需要 说 明 的 是 ，define 方法 是 PHP 定义 常量 的 方法 ， 只 有 通过 这 个 方法 才能 
在 PHP 中 定义 常量 。 这 个 方法 带 有 两 个 参数 : 一 个 为 常量 名 ， 另 一 个 为 常量 值 。 对 于 常量 
名 ， 请 务必 使 用 一 对 双 引 号 将 其 包 里 起来。 对 于 常量 值 ， 请 遵循 如 下 规则 : 

口 若 常 量 值 为 字符 串 ， 务 必 使 用 一 对 双 引 号 将 其 包 里 起 来 。 
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口 若 常 量 值 为 数字 ， 则 无 需 使 用 双 引 号 。 

下 面 ， 来 看 看 如 何 使 用 define 方法 把 在 4.1 节 的 前 言 中 提 到 例子 写成 脚本 ， 如 例 4.2 
所 示 。 

【 例 4.2】 定义 常量 WEATHER。 


1 define ("WEATHER", "Sunny"); 


在 例 4.2 中 ，WEATHER 就 是 常量 名 ， 而 Sunny 是 常量 值 。 无 论 在 何 处 引用 该 常量 ， 
结果 都 是 一 样 的 。 

需要 进一步 说 明 的 是 , 常量 名 最 好 能 很 好 地 说 明 该 常量 指 代 的 意义 。 同 时 ， 还 要 注意 ， 
常量 名 的 每 一 个 字母 最 好 都 大 写 ， 这 样 可 以 很 好 的 与 变量 区 分 开 。 当 然 ， 无 论 是 给 常量 取 
一 个 无 意义 的 名 字 ， 还 是 全 部 用 小 写字 母 ， 对 于 PHP 引擎 来 说 都 是 一 样 的 。 其 实 ， 这 样 做 
并 不 是 给 PHP 引擎 看 的 ， 而 是 给 我 们 自己 看 的 。 因此， 为 了 更 好 地 阅读 代码 ， 请 务必 使 用 
具有 描述 性 的 词语 给 常量 取 名 ， 并 大 写 每 一 个 字母 。 

在 例 4.1 中 ， 我 们 说 到 ， 如 果 常 量 值 为 数字 时 ， 常 量 值 前 后 无 须 使 用 双 引 号 。 现 在 我 
们 来 定义 一 个 常量 INTEREST_RATE， 其 值 为 0.652， 如 例 4.3 所 示 。 

【 例 4.3】 定义 一 个 数字 类 型 的 常量 。 


1 define ("INTEREST RATE", 0.652); 


说 到 这 里 ， 大 家 应 该 基本 掌握 了 如 何 定义 常量 。 那 么 ， 在 定义 常量 时 ， 除 了 使 用 具有 
描述 性 的 词语 给 常量 命名 以 外 ， 还 有 其 他 需要 注意 的 吗 ? 答案 当然 是 肯定 的 。 在 定义 常量 
时 ， 千 万 不 要 使 用 PHP 的 命令 关键 字 来 给 常量 命令 。 否 则 ，PHP 引擎 会 因为 无 法 分 辨 常量 
与 命令 关键 字 而 报错 。 来 看 一 个 例子 。 

【 例 4.4】 用 命令 关键 字 echo 来 定义 一 个 常量 。 


站 <?php 

他 define ("ECHO", "Hello Worldl!") 
3 echo ECHO; 

4 ?> 


将 例 4.4 所 示 的 脚本 保存 为 PHP 脚本 文件 并 加 以 执行 ， 会 有 什么 结果 呢 ? 图 4-1 所 示 
展示 了 运行 该 脚本 后 的 结果 : 


jr 
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Tee) 


将 Favorites 。 镍 http://www.greatwalltea/Ex4-4.php 丛 ” 国 "局 唤 ~pPagev Safetyv Toosv 加 


Parse error: syntax error, unexpected ECHO’ (T_ECHO) in C:\greatwall\Ex4-4.php on line 3 


图 4-1 使 用 命令 关键 字 定义 常量 的 系统 报错 


根据 系统 提示 ， 脚 本 出 现 了 语法 错误 。 在 脚本 的 第 三 行 出 现 了 一 个 未 知 的 ECHO。 这 
就 表明 了 PHP 引擎 无 法 区 分 例 4.4 所 示 脚 本 中 的 两 个 echo。 
这 样 看 来 ， 在 给 常量 取 名 时 ， 务 必要 避 开 系统 定义 的 命令 关键 字 。 表 4-1 所 示 列 出 了 
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表 4-1 常用 的 PHP 命 令 关键 字 


序号 命令 关键 字 命令 关键 字 
1 and 21 list 
2 as new 
3 break or 
4 Case print 
5 class Tequire 
6 const for 2 Teturm 
7 continue switch 
8 decline use 
9 die 下 29 Var 
10 do while 


4.1.2” 何 时 使 用 常量 


当 某 个 值 需要 在 脚本 运行 过 程 中 恒定 不 变 ， 就 可 以 将 其 定义 为 一 个 常量 。 使 用 常量 的 


好 处 是 显而易见 的 。 通 过 定义 常量 ， 可 以 用 具有 描述 性 的 常量 名 来 替代 常量 值 ， 从 而 使 得 
代码 维护 更 加 容易 。 例 如 ，INTEREST_RATE 显然 要 比 0.652 好 懂 的 多 。 


除 此 之 外 ， 当 定义 了 一 个 常量 之 后 ， 可 以 在 脚本 的 任何 地 方 引 用 该 常量 。 当 修改 了 该 
常量 的 值 ， 所 有 引用 该 常量 的 地 方 也 会 同步 更 新 修改 后 的 常量 值 。 因 此 ， 只 需 更 新 一 处 代 
码 ， 即 可 实现 脚本 全 局 的 更 新 。 


下 面 还 是 用 汇率 来 举例 。 假 设 美元 对 人 民 币 的 汇率 为 6.22， 我 们 可 以 写 下 如 例 4.5 所 
示 的 脚本 。 

【 例 4.5】 定义 汇率 为 常量 。 

1 <?php 

4 define ("INTEREST RATE", 6.22); 

3 $USD = 100; 

4 SRMB = $USD * INTEREST RATE; 

5 echo $RMB; 

6 ?> 


在 上 面 这 个 例子 中 ， 我 们 定义 了 一 个 常量 INTEREST_ RATE， 其 值 为 6.22。 它 代表 了 
美元 兑 人 民 币 的 汇率 。 我 们 现在 想 知道 手中 的 100 美元 相当 于 多 少 人 民 币 。 其 值 应 该 是 拥 
有 的 美元 数量 乘 以 当前 利率 ， 为 622。 为 了 计算 这 个 结果 ， 我 们 定义 了 两 个 变量 :一 个 为 
美元 (SUSD) ， 另 一 个 为 人 民 币 (SRMB) 。 需 要 注意 的 是 ， 在 定义 变量 时 ， 变 量 名 前 要 
加 上 变量 定义 符 ($) 。 

然后 ， 我 们 为 变量 SUSD 赋值 100， 表 示 我 们 拥有 的 100 美元 。 按 照 之 前 的 思路 ， 变 
量 SRMB 的 值 应 该 为 SUSD 乘 以 INTEREST RATE。 于 是 ， 我 们 为 SRMB 赋值 SUSD * 
INTEREST RATE。 

最 后 打印 变量 SRMB 的 值 。 

例 4.5 中 的 这 段 脚 本 只 有 六 行 ， 看 上 去 定义 常量 的 作用 不 大 。 因 为 脚本 只 在 第 四 行 引 
用 了 该 常量 。 但 是 通常 脚本 不 会 只 有 六 行 ， 常 量 的 引用 也 不 会 只 有 一 处 。 假 设 脚 本 有 上 和 干 
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行 ， 对 该 常量 的 引用 有 数 十 处 ， 定 义 常量 的 意义 就 显而易见 了 。 
有 的 同学 可 能 会 问 ， 可 不 可 以 把 利率 定义 为 一 个 变量 呢 ， 比 如 写成 如 例 4.6 所 示 的 


脚本 。 
【 例 4.6】 定义 汇率 为 变量 。 
是 <?php 
2 S$rate = 6.52; 
3 $USD = 100; 
二 SRMB = SUSD * S$rate; 
5 echo &RMB; 
SS 


例 4.5 与 例 4.6 的 区 别 就 在 于 前 者 定义 了 一 个 常量 INTEREST_RATE， 并 在 计算 人 民 
币 价值 时 引用 了 该 常量 ， 后 者 则 定义 了 一 个 变量 $rate， 然 后 在 计算 人 民 币 价值 时 引用 了 该 
变量 。 两 者 的 其 他 代码 均 相 同 ， 得 出 的 结果 也 是 相同 的 。 但 是 刚才 也 提 到 了 ， 通 常 编写 的 
代码 肯定 不 止 六 行 ， 假 设 在 茫茫 数 千 行 代码 中 的 某 处 ， 我 们 写 下 了 如 例 4.7 所 示 的 代码 。 
【 例 4.7】 重复 定义 变量 Srate。 


11 S$rate = 80.31 


35 S$RMB = $USD * S$rate; 
36 echo &RMB; 


在 例 4.7 中 ， 我 们 在 第 11 行 重复 定义 了 Srate， 并 为 其 赋 了 一 个 新 值 80.31。 之 后 ,在 
第 35 行 又 一 次 引用 了 变量 $rate。 这 时 ， 计 算出 的 SRMB 的 值 就 变 成 了 8031， 而 不 是 之 前 
的 622 了 。 而 且 在 这 行 代码 之 后 所 有 引用 Srate 变量 的 地 方 ， 都 会 更 新 成 80.31， 并 被 代入 
计算 。 

即便 十 分 小 心 ， 也 难保 不 会 出 现 这 样 的 错误 。 所以， 对 于 一 个 短期 内 恒定 不 变 或 波动 
很 小 的 值 ， 将 其 定义 为 常量 ， 是 一 个 不 错 的 选择 。 


4.1.3 ”PHP 预 置 常 量 


PHP 也 为 我 们 预定 义 了 一 些 常量 。 比 如 常量 _LINE 表示 该 常量 所 在 的 行 号 ， 常 量 
_FILE 表示 该 常量 所 在 的 文件 名 。 这 两 个 常量 生得 有 些 奇 怪 : 以 两 个 下 划 线 开始 ， 再 以 
两 个 下 划 线 结束 。 大 家 可 以 通过 如 例 4.8 所 示 的 脚本 来 检验 这 两 个 常量 。 

【 例 4.8】 常量 _ LINE 和 _FILE 。 


天 <?php 
echo "Constant LINE has been used in line ", LINE ,".<br>"; 
echo "The current file locates in ", FILE ,"."; 


OD 


在 例 4.8 所 示 的 脚本 使 用 echo 命令 输出 了 两 句 话 。 第 一 句 话说 的 是 常量 _LINE_ 出现 
在 脚本 的 第 几 行 ， 第 二 句 话说 的 是 当前 文件 存放 在 什么 路 径 下 。 我 们 运行 该 脚本 可 以 得 出 
如 图 4-2 所 示 的 结果 。 

根据 截图 中 的 内 容 , 大 家 可 以 知道 常量 _LINE_ 所 在 的 行 和 常量 _FILE_ 所 在 文件 的 
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存放 路 径 。 
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Constant _LIIE_ has been used in line 2. 
The current file locates in C:\greatwall\index. php. 


图 4-2 常量 LINE 和 FIIE _ 


除了 上 述 两 个 预 置 常量 之 外 , PHP 还 提供 了 一 些 用 于 处 理 脚 本 错误 的 常量 , 如 E_ALL 
和 ERROR 等 。 这 些 常量 可 以 影响 PHP 处 理 引擎 如 何 处 理 脚 本 中 出 现 的 错误 。 


4.2 什么 是 变量 


在 上 一 节 例 4.5 中 ， 我 们 曾 提 到 定义 变量 需要 使 用 变量 定义 符 〈$) 。 这 也 说 明了 变量 
的 定义 方法 与 常量 有 着 较 大 的 区 别 。 按 照 PHP 脚本 语言 的 语法 规则 , 常量 前 不 加 任何 符号 
而 变量 前 则 必须 带 变量 定义 符 。 在 本 节 里 ， 我 们 就 深入 讨论 一 下 变量 。 


4.2.1 变量 的 命名 


现在 大 多 数 网 站 会 要 求 大 家 注册 ， 在 注册 时 ， 需 要 填写 诸多 的 信息 。 注 册 完 毕 后 ， 大 
家 使 用 注册 的 用 户 名 和 密码 登录 网 站 ， 会 发 现 网 站 某 个 部 分 显示 了 我 们 注册 的 用 户 名 。 这 
一 点 相信 大 家 并 不 陌生 。 

其 实用 户 注 册 信息 是 通过 变量 在 数据 库 和 客户 端 之 间 传递 的 。 在 PHP 脚本 语言 里 ， 变 
量 通常 用 来 保存 用 户 填写 在 表单 里 诸如 姓名 、 年 龄 和 性 别 之 类 的 信息 。 之 后 ， 可 以 通过 引 
用 相应 的 变量 来 定制 页 面 的 内 容 。 这 样 一 来 ， 每 个 用 户 看 到 的 都 是 打上 了 不 同 个 人 标记 的 
页 面 。 

在 了 解 如 何 定义 变量 之 前 ， 先 来 看 看 变量 的 命名 规则 。 

和 常量 类 似 ， 变 量 也 有 变量 名 和 变量 值 。 不 过 在 定义 变量 的 时 候 ， 不 需要 使 用 define 
方法 。 与 定义 常量 比较 起 来 ， 定 义 变量 有 如 下 几 点 要 求 : 

口 所 有 的 变量 名 都 必须 以 变量 定义 符 〈$) 开头 。 变 量 定义 符 告诉 PHP 处 理 引擎 接 下 

来 要 处 理 一 个 变量 。 

口 变量 名 的 长 度 没有 限制 ， 不 过 建议 大 家 在 为 变量 命名 时 使 用 具有 描述 性 的 词语 。 
这 一 点 要 求 与 常量 相同 。 
变量 名 只 能 使 用 字母 、 数 字 和 下 划 线 。 其 他 类 型 的 字符 一 律 不 能 使 用 。 
变量 定义 符 后 的 第 一 个 字符 必须 为 字母 或 下 划 线 ， 不 能 使 用 数字 。 
使 用 拼写 相同 的 单词 的 大 写 形式 和 小 写 形 式 定义 变量 时 ， 得 到 的 是 两 个 不 同 的 变 
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量 。 比 如 $Car Model 和 $car model 是 两 个 不 同 的 变量 。 
表 4-2 所 示 列 出 了 一 些 合法 的 与 非法 的 变量 名 ， 大 家 可 以 参照 上 面 的 规则 来 分 析 一 
合法 的 为 什么 合法 ， 非 法 的 又 为 什么 非法 。 


表 4-2 变量 名 举例 


| Sfirst_ name 
| 


$name 3 $first-name 


$salute to user 


重点 关注 一 下 表 4-2 中 列 出 的 非法 变量 名 。 这 些 变 量 名 之 所 以 是 非法 的 ， 要 么 是 因为 
在 变量 定义 符 后 的 第 一 个 字符 处 使 用 了 数字 ， 要 么 在 变量 名 中 使 用 了 非法 字符 。 使 用 这 些 
变量 名 ， 系 统 会 报错 。 以 $firsttname 为 例 来 看 一 下 PHP 引擎 在 磁 到 这 个 变量 名 时 的 反应 ， 
如 图 4-3 所 示 。 


$salute_to_user $key note 


Sa El 
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Notice: Use of undefined constant name - assumed "name” in C:\greatwall\Table4-2.php on 
| line 2 


Notice: Undefined variable: first in C:\greatwall\Table4-2.php on line 2 


图 4-3 ”关于 非法 变量 名 $firsttname 的 报错 信息 


根据 截图 中 的 内 容 ， 可 以 发 现 PHP 处 理 引擎 将 name 当成 了 一 个 未 定义 的 常量 ， 而 将 
该 非法 变量 名 中 的 “+” 当 成 了 运算 符 ， 进 而 认定 $first 是 一 个 未 定义 的 变量 。 于 是 出 现 了 
图 4-3 中 所 示 的 两 条 报错 信息 

另外 , 在 表 4- 2 所 示 中 出 现 了 如 SwamMsg 和 $first_name 这 样 的 变量 名 , 它们 也 是 合法 
的 变量 名 。 由 于 在 PHP 脚本 中 ， 变 量 名 的 长 度 是 没有 限制 的 ， 因 此 可 以 用 任何 想 用 的 字母 
组 合 来 给 变量 命名 。 但 是 有 一 点 , 不 能 在 变量 名 中 使 用 除了 下 划 线 以 外 的 其 他 符号 。 因 此 ， 
如 果 想 用 两 个 或 者 更 多 单词 来 给 变量 命名 时 ， 要 么 在 这 些 单词 间 使 用 下 划 线 ， 要 么 把 从 第 
二 个 单词 开始 的 每 个 单词 的 首 字母 大 写 。 在 本 书 中 变量 名 的 命名 采用 第 二 种 方式 。 

这 样 做 的 理由 同样 也 是 为 了 方便 今后 阅读 和 维护 脚本 。 读 者 也 可 以 不 这 样 做 ， 不 过 在 
维护 脚本 时 可 能 要 费 一 些 功夫 来 辨别 每 个 变量 代表 的 意思 。 


4.2.2 如何 定 义 变量 


在 了 解 了 变量 的 命名 规则 之 后 , 再 来 看 看 如 何 定义 一 个 变量 。 在 讲 变量 的 命名 规则 时 ， 
我 们 提 到 过 定义 变量 是 不 需要 使 用 define 方 法 的 .那么 在 PHP 中 应 该 如 何 定 义 一 个 变量 呢 。 
有 一 句 话 是 这 么 说 的 : 为 变量 赋值 即 定义 一 个 变量 。 也 就 是 说 如 果 已 经 给 一 个 变量 分 配 了 
对 应 的 变量 值 ， 那 么 这 个 变量 是 否 存在 就 不 再 是 个 问题 。 
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在 PHP 中 ， 赋 值 用 的 是 等 号 (=) ， 叫 做 赋值 符 ， 意 思 是 把 赋值 符 右边 的 值 分 配给 赋 
值 符 左 边 的 变量 。 赋 值 符 左 右 的 内 容 不 能 互 换 ， 这 与 数学 中 的 等 号 概念 不 同 。 例 4.9 所 示 
展示 了 如 何 为 变量 赋值 。 

【 例 4.9】 为 变量 赋值 。 

1 S$name = "David Clark"; 

2 $age = 237 

3 S$profession = "Assistance"; 

4 $salary = 8904.34; 

在 例 4.9 中 定义 了 四 个 变量 ， 有 的 变量 值 前 后 带 有 一 对 引号 ， 有 的 则 没有 。 带 有 引号 
的 是 字符 串 类 型 的 变量 。 这 一 对 引号 告诉 PHP 处 理 引 擎 将 引号 内 的 全 部 内 容 当 成 一 个 值 赋 
值 给 赋值 符 左边 的 变量 。 不 带 引 号 的 是 数字 类 型 的 变量 。PHP 处 理 引擎 在 碰 到 这 种 类 型 的 
变量 值 时 , 会 直接 将 其 赋值 给 赋值 符 左 边 的 变量 。 值 得 注意 的 是 , 无 论 是 字符 串 型 的 变量 ， 
还 是 数字 类 型 的 变量 ， 在 PHP 脚本 中 ， 都 是 可 以 计算 的 。 

变量 除了 有 不 同 的 类 型 ， 还 可 以 被 重复 定义 。 这 一 点 在 定义 常量 时 是 无 法 实现 的 。 比 
如 ， 可 以 运行 一 下 例 4.10 所 示 的 脚本 。 

【 例 4.10】 重复 定义 一 个 常量 。 


1 <?php 

受 define ("AGE", 19); 
到 define ("AGE", 20); 
ee 


运行 后 ,系统 会 报错 。 报错 信息 如 图 4-4 所 示 , 常量 AGE 已 经 定义 了 , 不 能 再 次 定义 。 
如 果 需 要 修改 常量 的 值 ， 则 务必 在 定义 该 常量 的 语句 中 修改 。 


em im =m leo ee 


p/wwgre 
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Notice: Constant AGE already defined in C:\greatwall\Ex4-10.php on line 3 


图 4-4 重复 定义 常量 时 的 报错 信息 


对 于 变量 而 言 ， 重 复 定 义 是 十 分 普遍 的 。 再 来 看 一 段 脚 本 ， 如 例 4.11 所 示 。 
【 例 4.11】 重复 定义 一 个 变量 。 

主 <?php 

$color = "blue"; 


$color = "red"; 
echo "The color is $color"; 


LOD 


2> 

这 段 脚本 的 运行 结果 输出 了 第 二 次 我 们 为 变量 $color 赋 的 值 。 这 也 说 明 ， 变 量 是 可 以 
被 重复 定义 的 。 需 要 注意 的 是 ， 一 旦 再 次 定义 了 某 个 变量 ， 该 变量 的 值 就 发 生 了 变化 ， 之 
前 为 其 赋 的 值 就 被 最 新 赋 的 值 荐 盖 了 。 
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值得 注意 的 是 ， 在 这 段 脚本 的 第 四 行 ， 我 们 把 变量 名 Scolor 放 在 了 引号 里 面 ， 这 样 它 
就 可 以 和 引号 中 的 其 他 信息 一 起 输出 。 但 是 把 变量 名 写 在 引号 里 面 有 时 也 会 存在 问题 。 下 
面 来 看 一 个 例子 。 

【 例 4.12】 在 引号 中 使 用 变量 (一 ) 。 


工 <?php 

$bodyPart = "tooth"; 

echo "You have a $bodyPartache."™; 
Ea 


心 w N 


其 实 脚本 的 初衷 是 想 让 PHP 处 理 引 擎 输出 “You have a toothache.” 这 人 句 话 ， 结 果 它 给 
了 我 们 一 记 白 眼 ， 输 出 了 下 面 这 名 话 : 

Notice: Undefined variable: bodyPartache in C:\greatwall\Ex4-12.php on line 3 

原来 PHP 处 理 引 擎 把 bodyPartache 当成 了 一 个 变量 ， 而 脚本 中 又 没有 定义 这 个 叫 
bodyPartache 的 变量 ， 所 以 出 现 了 错误 。 那 么 有 没有 办 法 可 以 修改 一 下 呢 ? 答案 是 肯定 的 。 
其 实 ， 在 引号 中 使 用 的 变量 前 后 加 上 一 对 花 括号 人) 就 可 以 实现 目的 。 因此， 可 以 把 例 
4.12 修改 成 如 下 的 样子 : 

【 例 4.13】 在 引号 中 使 用 变量 (二 ) 。 


. <?php 
$bodyPart = "tooth"; 
echo "You have a {$bodyPart}ache."; 


SON 


2 


还 可 以 把 一 个 变量 做 为 值 赋 给 另 一 个 变量 。 这 一 点 也 是 常量 无 法 做 到 的 ， 如 例 4.14 
所 示 。 
【 例 4.14】 把 变量 做 值 赋 给 另 一 个 变量 。 


1 <?php 

2 Suser = "Sally"; 

3 $salute = "Good Morning! "7 

4 $saluteToUser = $salute.$user; 
过 echo $saluteToUser; 

6 3> 


这 段 脚本 定义 了 三 个 变量 ， 分 别 为 Suser、$salute 和 $salueToUser。 其 中 $saluteToUser 
的 值 为 变量 $salute 和 $user 的 组 合 。 输 出 的 结果 就 是 这 两 个 变量 的 值 的 组 合 。 


4.2.3 详 谈 变 量 输出 


到 这 里 , 我 们 已 经 使 用 echo 命令 编写 了 几 个 脚本 输出 了 一 些 变量 的 值 。 可 能 对 于 某 些 
同学 来 说 ， 通 过 这 些 脚 本 掌握 的 知识 是 零散 的 。 为 了 巩固 对 输出 变量 的 认识 ， 下 面 把 可 能 
会 见 到 的 变量 输出 方式 做 个 总 结 。 

表 4-3 所 示 中 使 用 的 变量 定义 如 下 : 


Snumber = 159; 
$wordl1 = "Hello"; 
Sword2 = "World™; 
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表 4-3 使 用 echo 命 令 输出 变量 及 其 结果 示例 


Echo 命令 输出 结果 
echo $number; 159 
echo $word],$word2; HelloWorld 
echo $wordl1," ",$word2; Hello World 


echo 'Say "$word]1 $word2" now'! Say "Hello World" now! 


说 到 变量 的 输出 ， 再 喝 吴 两 句 。 其 实 PHP 提供 了 不 止 echo 一 种 输出 信息 的 方法 。 还 
有 若干 种 其 他 的 输出 信息 的 方法 。 其 中 一 种 叫做 print 方法 ， 它 的 功能 是 输出 关于 变量 易 
于 人 理解 的 信息 。 对 于 单个 变量 来 说 ， 无 论 该 变量 是 字符 串 、 整 型 还 是 浮 点 数 变 量 ， 使 用 
print T 方法 都 可 以 方便 地 输入 该 变量 的 值 ， 这 一 点 与 echo 命令 一 样 。 但 是 与 echo 命令 不 
同 的 是 ，print r 一 次 只 能 输出 一 个 变量 的 值 ， 而 echo 一 次 可 以 输出 多 个 变量 的 值 。 为 了 说 
清楚 这 个 问题 ， 来 看 一 段 脚本 。 

【 例 4.1S】 print r 和 echo 的 区 别 。 


h! <?php 

$user = "Sally"; 

$salute = "Good Morning! "7 
print r($user); 

print r($salute); 

echo $salute, $user; 


-wmwwN 


2% 


例 4.15 使 用 print Ir 方法 输出 了 $user 和 $salute 两 个 字符 串 变 量 的 值 ， 分 别 为 “Sally” 
和 “Good Morning!”。 然 后 使 用 echo 方法 输出 了 这 两 个 变量 值 组 合 在 一 起 的 结果 。 

使 用 echo 命令 输出 变量 值 的 时 候 ， 如 果 引 用 未 定义 的 变量 ， 系 统 会 报错 。 比 如 脚本 中 
定义 了 一 个 变量 $age， 结 果 在 echo 命令 中 引用 时 写成 了 $aeg。 这 时 系统 会 出 现 如 下 所 示 的 
错误 。 


Notice: Undefined variable: aeg in C:\testvar.php on line 5 


读者 可 以 通过 在 变量 引用 的 变量 名 前 加 @ 来 关闭 系统 报错 (如 echo @$aeg) 。 这 样 系 
统 就 不 会 显示 上 述 出 错 信 息 。 但 是 由 于 $aeg 的 确 是 一 个 未 定义 的 变量 ， 所 以 系统 也 不 会 输 
出 任何 信息 ， 你 也 不 会 看 到 任何 结果 。 

此 处 的 建议 是 ， 如 无 必要 ， 千 万 不 要 关闭 任何 出 错 信 息 。 不 是 有 和 句 话 这 么 说 的 么 : 出 
了 错 不 可 怕 ， 可 怕 的 是 出 了 错 自己 都 不 知道 错 在 哪儿 。 

知 错 能 改 ， 善 莫大 淖 。 


4.2.4” 何 时 使 用 变量 


在 知道 如 何 定义 变量 和 输出 变量 之 后 , 还 需要 知道 何 时 使 用 变量 。 与 常量 的 使 用 类 似 ， 
当 某 种 信息 会 由 于 条 件 的 不 同 而 发 生变 化 ， 那 么 使 用 变量 是 最 合适 不 过 的 了 。 例 如 ，“ 天 
气 ” 通 常 被 看 成 一 个 复杂 多 变 的 概念 ， 人 们 的 活动 会 随 着 天 气 的 不 同 而 发 生变 化 。 那 么 ， 
可 以 编写 一 段 PHP 脚本 来 描述 这 种 变化 。 

【 例 4.16】 示例 脚本 : 人 与 天 气 。 


<?php 
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城 让 


// 定 义 变量 $weather 


Sweather = "sunny"; 


// 描 述 在 不 同 天 气 条 件 下 人 的 活动 方式 
Switch (Sweather) { 
case "sunny™ : 
echo "It's ", $weather,", you can take a walk in the sunset."; 
break; 
case "rainy™" : 
echo "It's ", $weather,", you should take an umbrella with you."; 
break; 
case "windy" : 
echo "It's ",$weather,", you should wear your jacket."; 
break; 


} 


例 4.16 定义 了 一 个 变量 $weather， 并 且 描述 了 在 不 同 天 气 情 况 下 ， 人 们 一 般 会 从 事 的 
活动 。 通 过 修改 变量 $weather 的 值 ， 脚 本 会 输出 不 同 的 描述 片段 。 这 里 可 以 使 用 switch 方 
法 。 它 和 第 2 章 里 见 到 的 站 命令 类 似 ， 都 是 用 于 描述 条 件 的 。 翻 译 成 自然 语言 就 是 ， 如 果 
出 现 了 A 状况 ， 如 何如 何 ; 如 果 出 现 了 B 状况 ， 如 何如 何 。 关 于 switch 方法 ,我 们 会 在 第 
8 章 中 详细 讲 到 。 

在 PHP 脚本 中 ， 使 用 变量 的 情况 要 比 使 用 常量 的 情况 多 的 多 。 可 以 说 大 多 数 情况 下 ， 
都 需要 使 用 变量 。 这 不 光 是 因为 变量 定义 起 来 很 方便 ， 还 因为 与 常量 比较 起 来 ， 变 量 更 加 
灵活 。 再 来 看 一 个 例子 。 

【 例 4.17】 用 变量 A 的 值 充当 变量 B 的 名 :显示 相关 城市 人 口 数量 〈 单 位 : 万 人 ) 。 


<?php 


3 


// 使 用 城市 名 称 定义 若干 变量 ， 用 于 存储 各 城市 的 人 口 数量 
$shanghai = 2231.5; 

Sbeijing = 1882.7; 

$chongqin = 1569.4; 

Stianjin = 1109.0; 

$guangzhou = 1107.1; 


// 定 义 一 个 变量 用 于 存储 城市 名 称 


$cityName = "beijing"; 


// 输 出 北京 市 的 人 口 数量 
echo "The population in S$cityName is ${$cityName} .<br>"; 


// 修 改 $cityName 的 值 为 广州 市 


ScityName = "guangzhou"; 


// 输 出 广州 市 的 人 口 数量 


echo "The population in $cityName is ${$cityName}."; 


例 4.17 输出 的 结果 如 图 4-5 所 示 。 
例 4.17 分 别 用 五 个 城市 的 名 字 定 义 了 五 个 变量 ,用 来 存储 这 五 个 城市 的 人 口 数量 。 然 
后 又 定义 了 一 个 变量 用 来 指定 这 五 个 城市 中 的 某 一 个 城市 , 最 后 使 用 echo 命令 输出 关于 该 


的 人 口 数 量 的 一 句 描述 。 


大 家 可 以 看 到 例子 中 有 两 条 echo 命令 。 它 们 都 是 一 样 的 。 
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[en el] 


bs DE 5 


| 这 Favorites 。 臣 http://www.greatwalltea/Ex4-17.php 顺 ”pagev Safetyv Toosv 轿 ” 


The population in beijing is 1882.7. 
| The population in guangzhou is 1107.1. 


图 4-5 用 变量 A 的 值 充当 变量 B 的 名 : 输出 结果 
echo "The population in ScityName is ${$cityName} .<br>"; 


在 这 条 echo 命令 中 ， 变 量 $cityName 使 用 了 两 次 。 但 是 根据 输出 的 结果 来 看 ， 第 一 次 
输出 了 城市 的 名 字 ， 而 第 二 次 却 输出 了 城市 的 人 口 数 量 。 这 是 为 什么 呢 ? 原来 在 第 二 次 引 
用 变量 $cityName 时 ， 在 它 的 前 面 又 加 了 一 个 变量 定义 符 〈$) 。 当 指定 变量 $cityName 的 
值 为 beijing 时 , $$cityName 就 等 价 于 $beijing。 当 指定 变量 $cityName 的 值 为 guangzhou 时 ， 
$8cityName 就 等 价 于 $guangzhou。 因 为 PHP 处 理 引 擎 在 磁 到 $$cityName 时 , 会 认为 两 个 变 
量 定义 符 ($) 后 面 跟着 的 都 是 变量 名 ， 于 是 把 $cityName 转换 成 了 它 对 应 的 值 ， 然 后 又 把 
以 $cityName 对 应 的 值 为 变量 名 的 变量 转换 成 了 它 对 应 的 值 。 

在 这 个 例子 中 ， 正 是 有 了 这 样 的 两 次 转换 ， 才 在 两 次 引用 同一 变量 的 情况 下 ， 输 出 了 
不 同 的 值 。 


4.2.5 ”如何 销 毁 变 量 
在 定义 了 某 个 变量 之 后 ， 如 果 一 旦 不 需要 这 个 变量 了 ， 可 以 销毁 它 。 来 看 下 面 这 个 


例子 。 
【 例 4.18】 销毁 变量 。 


全 <?php 

// 定 义 一 个 变量 $age 
3 S$age = 10; 

4 

5 // 输 出 变量 $age 

6 echo S$age; 

沉 

8 // 销 毁 变 量 Sage 

9 unset ($age); 

10 

了 // 再 次 输出 变量 Sage 
EL echo $age; 

i130 > 


大 家 猜测 一 下 ， 在 运行 这 段 脚本 后 ， 浏 览 器 会 有 什么 反应 呢 ? 图 4-6 所 示 展 示 了 浏览 
器 的 反应 。 

根据 图 4-6 显示 的 结果 ， 当 脚本 第 一 次 要 求 输出 变量 $age 的 时 候 ，PHP 处 理 引 擎 照 做 
了 。 但 是 当 使 用 unset 方法 销毁 变量 gage 后 再 次 输出 时 ，PHP 处 理 引 擎 又 给 了 我 们 一 记 白 
眼 ， 输 出 了 错误 提示 : 
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| 


© 9 Cry 加 四 四 


次 Fmente | 车 hapyjnvwereatvallicayB438php 位 "四 ”名 -pas Say Tookv 司 - 


10 
NMotice: Undefined variable: age in C:\greatwall\Ex4-18. php on line 5 


图 4-6 销毁 变量 


Notice: Undefined variable: age in C:\greatwall\Ex4-18.php on line 5. 


这 条 错误 提示 说 明 变 量 $age 此 时 已 经 不 复 存 在 了 。 
4.3 实战 练习 : 常量 与 变量 


4.3.1 背景 介绍 


本 节 将 使 用 本 章 中 学 到 的 知识 来 编写 一 个 货币 兑换 的 脚本 。 在 这 里 ， 以 人 民 币 为 基础 
货币 , 同时 以 2013 年 7 月 5 日 单位 人 民 币 兑换 各 主要 货币 的 汇率 为 常量 , 开展 货币 兑换 的 
业务 ， 如 图 4-7 所 示 。 


今日 处 汇 牌 位 最 新 查询 《2013 年 7 月 5 日 ) 

币 种 交易 单位 中 间 价 现汇 买 入 价 现 抄 买 入 价 实 出 价 
元 nsD) 100 | mm | li 加 到 站 
洪 而 00) 1o0 | 的 me 的 TB.28 915 
欧元 EUR) 10 | Tm9% 706.78 T61 90 793.10 
英 涝 pp i109 | sl 和 GIT.18 ee 1 5 14 
日 元 CD 100 | 6.10 6.08 5.69 B13 


图 4-7 2013 年 7 月 5 日 外 汇 牌价 


货币 兑换 平台 看 上 去 如 图 4-8 所 示 的 样子 。 为 此 ， 需 要 先 用 HIML 对 货币 兑换 平台 
行 装修 。 


图 4-8 货币 兑换 平台 
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HTML 代码 如 下 : 
<!DOCTYPE html> 
<html> 
<head> 
<meta charset="UTF-8"> 
<title> 外 汇兑 换 </title> 
<style type="text/css"> 
#msg { 
line-height:150px; 
text-align:center; 
font-size:230%; 
font-family:sans-serif; 
} 
form, h2 { 
text-align:center; 
} 
</style> 
</head> 
<body> 
<h2> 外 汇兑 换 </h2> 
<hr /> 


// 货 币 兑换 处 的 标题 


<form action='?check' method="post"> 


<input type='text' name="quantity"> // 用 户 输入 指定 外 汇 数量 的 文本 框 


<select name="currency"> 
<option 
<option 
<option 
<option 
<option 
</select> 
<button type="submit"> 竞 换 < 
</form> 
<div id="msg"></div> 
</body> 
</html> 


// 用 户 选择 指定 外 汇 各 类 的 下 拉 框 
value="'USD' > 美元 (USD) </option> 
value="'HKD' > 港元 (HKD) </option> 
value="'EUR' > 欧元 (EUR) </option> 
value="'GBP' > 英镑 (GBP) </option> 
value='JPY'> 日 元 (JPY) </option> 


// 用 户 提交 表单 的 按钮 


/button> 


// 显 示 兑 换 结果 的 div 


这 段 HTML 代码 使 用 了 一 张 表 单 ,并 定义 了 一 个 显示 兑换 结果 的 DIV。 然 后 使 用 了 一 
段 简 单 的 CSS 代码 对 页 面 进 行 了 装修 。 值 得 注意 的 是 ,为 了 在 当前 页 面 处 理 用 户 通过 表单 
提交 的 信息 ， 表 单 的 action 属性 的 值 设置 成 了 “=?check”。 然 后 通过 PHP 脚本 根据 action 


属性 的 值 来 判断 用 户 是 否 提交 了 表单 。 


关于 页 面 间 值 的 传递 会 在 第 9 章 中 讲 到 ， 这 号 


接 下 来 ， 需 要 在 “</body>” 标 签 前 面 添加 一 段 PHP 脚本 ， 来 使 表单 交互 起 作 


有 就 先 忽 略 过 去 了 。 


| 


。 注 


意 ， 添 加 PHP 脚本 的 地 方 一 定 要 是 “</body>” 标 签 前 ， 否 则 是 看 不 到 脚本 执行 的 结果 的 。 


需要 添加 的 PHP 脚本 如 下 : 


<?php 
if (isset($ REQUEST['check']))t{ 
define('USD', 6.125); 
define('HKD', 0.789); 
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// 判 断 用 户 是 否 已 经 提交 了 
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> 


上 


define('EUR', 7.899); 
define('GBP', 9.214); 
define('JPY'，0.061); // 定 义 五 个 常量 分 别 对 应 五 种 主要 货币 兑 人 民 币 的 汇率 


S$quantity = intval($ REQUESTI['quantity']);// 获 取 用 户 需 兑换 的 外 汇 数量 
$currency = $ REQUEST['currency']; // 获 取 用 户 所 持 有 的 外 汇 种 类 


switch ($currency) // 根 据 获取 的 外 汇 种 类 ， 计 算 等 值 人 民 币 数量 
{ 
case "USD" : 
Samount = $quantity * USD; 
break; 
case "HKD" : 
Samount = S$quantity * HKD; 
break; 
case 'EUR': 
Samount = S$quantity * EUR; 
break; 
case "GBP' : 
Samount = S$quantity * GBP; 
break; 
case "JPY' : 
Samount = S$quantity * JPY; 
break; 


有 


echo '<script ge="javascript">document .getElementById("msg") 
.innerHTML=""'. 
S$quantity.' '.$currency.' = '.$amount.' RMB'. 


nhscript> // 输 出 最 后 结果 到 ID 为 msg 的 div 中 


这 段 脚 本 通过 $_ REQUEST 这 样 一 个 系统 预 置 的 常量 数组 来 判断 当前 页 面 是 否定 义 了 
-个 索引 为 “check” 的 数组 元 素 。 值 的 注意 的 是 ， 在 判断 的 时 候 使 用 了 isset0 函 数 ， 即 判 

断 系 统 中 是 否 设置 了 这 个 “$_REQUEST['check]” 数 组 元 素 。 若 设置 了 ， 则 表明 用 户 提交 
了 表单 ; 若 没 有 设置 ， 则 表明 用 户 没 有 提交 表单 。 

这 里 提 到 的 $ REQUEST 常量 数组 其 实 就 是 在 页 面 间 传递 数组 时 经 常用 到 的 一 个 数 
组 。 与 它 类 似 的 两 个 常量 数组 分 别 是 3_POST 和 S$GET。 关 于 这 三 个 系统 预 置 的 常量 数组 ， 
大 家 可 以 参见 第 9 章 的 内 容 。 

系统 在 检查 到 用 户 提交 了 表单 之 后 便 开始 执行 PHP 脚本 寺 语 名 中 的 内 容 。 在 过 语 句 
中 ， 脚 本 分 成 了 四 个 步骤 。 它 们 分 别 是 : 

(1) 定义 五 个 常量 ， 分 别 用 来 存储 五 种 货币 与 人 民 币 的 汇率 。 

(2) 获取 用 户 在 表单 中 填写 的 需要 兑换 的 货币 数量 和 货币 类 型 。 

(3) 根据 获取 到 的 货币 数量 和 货币 类 型 计算 可 兑换 的 人 民 币 数量 。 

(4) 将 计算 结果 输出 到 浏览 器 指定 位 置 。 


第 一 步 使 用 defineO 函 数 定义 了 五 个 常量 用 来 存储 五 种 货币 与 人 民 币 的 汇率 。 需 要 记 住 


的 是 , 通过 define0 函 数 定义 的 常量 的 值 在 整个 脚本 中 都 不 会 发 生 改变 , 该 常量 也 不 能 再 次 


第 二 步 继续 使 用 $ REQUEST 这 个 常量 数组 获取 用 户 通过 表单 提交 的 内 容 ， 其 中 
$_REQUEST['quantity'] 这 个 数组 元 素 存 储 着 用 户 在 表单 中 填写 的 外 币 数量 , 而 $ REQUEST 
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[currency] 这 个 数组 元 素 存储 着 的 则 是 用 户 在 表单 中 选择 的 货币 类 型 。 细 心 的 读者 可 能 
经 发 现 了 ， 这 两 个 数组 元 素 的 索引 正 是 表单 中 定义 的 文本 框 元 素 和 下 拉 框 元 素 的 “name” 
属性 的 值 。 由 于 获取 的 $_ REQUEST['quantity] 这 个 元 素 的 值 为 字符 串 类 型 的 值 ， 因 此 ， 脚 
本 使 用 了 int0 函 数 将 其 转换 成 一 个 整 型 的 值 ， 以 便 后 续 的 计算 。 


第 三 步 根据 获 取 和 


外汇 类 型 和 数量 ,计算 可 兑换 的 人 民 币 数量 。 在 这 里 需要 注意 的 是 ， 


脚本 使 用 了 switch 语句 根据 货币 类 型 来 控制 脚本 执行 的 顺序 。 即 当 货 币 类 型 为 “HKD”， 
执行 “case 'HKD':” 部 分 的 脚本 ， 并 在 见 到 “break;” 语 句 时 ， 退 出 switch 语句 。 通 过 这 种 


方式 ， 就 可 以 获取 基于 用 户 输入 的 信息 计算 的 可 兑换 的 人 民 币 数量 。 

第 四 步 使 用 echo 语句 输出 一 段 JavaScript 代码 。 这 段 代 码 将 之 前 的 计算 结果 按照 定义 
的 格式 输出 到 浏览 器 的 指定 位 置 上 。 

当 在 浏览 器 中 打开 这 个 页 面 ， 并 在 文本 框 中 输入 一 个 任意 的 正 数 ， 系 统 就 会 在 指定 的 
位 置 返回 指定 数量 的 外 汇 可 兑换 的 人 民 币 数量 。 

本 例 使 用 了 switch 语句 控制 脚本 的 执行 顺序 、 演 示 了 如 何 定义 常量 和 变量 ， 以 及 如 何 
使 用 它们 进行 运算 来 得 出 想 要 的 结果 。 关 于 switch 语句 的 介绍 ， 大 家 可 以 参见 第 7 章 的 


内 容 。 


44 习 是 


(1) 根据 脚本 中 的 注释 完成 任务 。 


<?ph 


p 
// 请 在 下 方 添加 脚本 中 缺失 的 部 分 


Stotal = RATE * Samount7 
echo S$total; 


> 


(2) 根据 脚本 中 的 注释 完成 任务 。 


// 请 在 下 方 添加 一 段 脚本 ， 使 echo 语句 输出 一 段 错误 提示 


echo $c; 
2 


(3) 请 编写 一 段 脚本 ， 输 出 脚本 文件 在 服务 器 上 的 路 径 。 
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在 上 一 章 里 ， 我 们 提 到 了 数据 类 型 这 个 概念 。 数 据 类 型 ， 顾 名 思 义 ， 应 该 就 是 数据 是 
什么 类 型 。 常 量 和 变量 都 可 以 用 来 存储 不 同类 型 的 数据 ， 不 同类 型 的 数据 可 以 做 不 同 的 事 
情 。 例 如 ， 可 以 把 都 是 数字 类 型 的 变量 加 起 来 求 和 ， 但 是 把 都 是 字符 串 类 型 的 变量 相 加 却 
没有 什么 意义 。 在 本 章 里 ， 我 们 要 讲 一 讲 PHP 支持 的 数据 类 型 以 及 它们 的 用 法 。 


5.1 概 述 


在 这 一 节 里 ， 主 要 介绍 各 种 数据 类 型 的 特点 ， 并 掌握 如 何 为 变量 指定 数据 类 型 。 
5.1.1 数据 全 家 福 


看 到 本 章 的 标题 ， 各 位 同学 应 该 也 能 推测 出 来 ， PHP 支持 的 数据 类 型 应 该 有 许多 种 。 

现在 我 们 按照 它们 的 特点 和 出 场 频率 逐一 介绍 一 下 : 

口 整 型 数据 是 五 兄弟 中 的 排头 老大 ， 在 数学 方面 天 资 不 错 ， 加 减 乘除 全 会 算 。 它 的 
能 力 根 据 安装 有 PHP 引擎 的 服务 器 使 用 的 操作 系统 的 不 同 会 发 生变 化 ,一 般 来 说 ， 
正 负 二 十 亿 范 围 内 的 四 则 运算 交 给 它 肯 定 没有 问题 。 

口 浮 点 型 数据 在 五 兄弟 中 排行 老 二 ， 在 数学 方面 有 着 惊人 的 天 赋 ， 它 会 算 各 种 复杂 
的 小 数 运算 。 正 弦 、 人 余弦、 正切 、 余 切 、 平 方 根 和 微 积分 什么 的 交 给 它 一 点 问题 
也 没有 。 虽 然 它 的 能 力也 会 根据 服务 器 使 用 的 操作 系统 的 不 同 而 发 生变 化 ， 不 过 
通常 情况 下 ， 输 出 指定 位 数 的 小 数值 ， 那 你 就 擎 好 吧 。 

口 字符 串 型 数据 在 五 兄弟 中 排行 老 三 ， 平 时 好 舞 个 文 、 弄 个 墨 。 诗 词 歌 贱 什么 的 ， 
只 要 教 给 它 的 ， 它 能 正 着 背 过 来 ， 倒 着 背 过 去 ， 还 不 带 一 个 错字 儿 的 。 真 真 儿 的 
一 个 学 富 五 车 的 翩翩 俏 公 子 。 

口 布尔 型 数据 在 五 兄弟 中 排行 老 四 ， 是 个 正 儿 八 经 的 法 官 。 它 最 常 念 奶 的 一 句 台 词 
就 是 “to be or not to be, that is a question”。 别 看 它 平时 和 哈姆雷特 一 样 神明 明 的 ， 
让 它 给 断 个 案子 什么 的 ， 它 一 看 一 个 准 儿 。 它 的 本 事 可 大 了 ， 只 要 搂 上 一 眼 ， 能 
立马 告诉 你 什么 是 真 、 什 么 是 假 、 什 么 是 对 、 什 么 是 错 。 

口 时 间 日 期 型 数据 在 五 兄弟 中 排行 老 丝 ， 也 没 其 他 别 的 爱好 ， 就 好 摆 个 钟 、 和 弄 个 表 。 
时 间 是 怎么 从 指 颖 中 溜 走 的 ， 它 了 然 于 胸 。 千 万 不 要 没事 儿 就 问 他 几 点 了 ， 它 的 
答案 会 把 你 烦 死 。 你 问 为 什么 ? 因为 它 告诉 你 的 时 间 有 零 有 整 ， 一 说 一 长 串 ， 半 
天 也 停 不 下 来 。 

有 了 这 五 位 兄弟 做 你 的 左 膀 右 臂 , 你 的 PHP 脚本 肯定 会 如 虎 添 翼 , 变 得 多 姿 多 彩 起 来 。 
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知道 它们 各 自 的 能 耐 之 后 ， 我 们 来 看 看 ， 如 何 把 它们 运用 到 脚本 里 。 
5.1.2 为 变量 指定 数据 类 型 


PHP 是 一 种 弱 类 型 的 脚本 。 也 就 是 说 ， 你 不 需要 刻意 地 为 变量 指定 数据 类 型 。 就 像 在 
上 一 章 中 所 做 的 那样 , 为 一 个 变量 赋 了 一 个 什么 类 型 的 值 , 这 个 变量 就 是 什么 类 型 的 变量 。 
因为 PHP 处 理 引 擎 可 以 根据 变量 的 值 自动 判断 该 变量 的 数据 类 型 , 同时 还 会 根据 实际 情况 
来 变更 数据 的 类 型 。 来 看 一 段 脚本 。 

【 例 $.1】 为 变量 指定 数据 类 型 。 


1 $firstNumber = 1; //PHP 处 理 引擎 自动 认定 该 变量 为 整 型 变量 

2 $secondNumber = 1.1; //PHP 处 理 引擎 自动 认定 该 变量 为 浮 点 型 变量 

3 S$thirdNumber = $firstNumber + $secondNumber; 

在 例 5.1 所 示 的 脚本 里 ， 定 义 了 两 个 变量 $firstNumber 和 $secondNumber， 并 为 


$firstNumber 赋 了 一 个 整 型 值 1， 为 $SsecondNumber 赋 了 一 个 浮 点 值 1.1。 然 后 定义 了 第 三 
个 变量 $thirdNumber， 用 来 存储 前 两 个 变量 的 和 。 要 是 放 在 其 他 的 编程 语言 里 ， 直 接 把 这 
两 个 不 同类 型 的 变量 直接 相 加 是 不 可 能 的 。 但 是 在 PHP 里 就 不 同 了 , PHP 处 理 引 擎 可 以 自 
动 将 整 型 变量 转换 成 浮 点 型 变量 ， 然 后 再 把 两 个 变量 相 加 以 后 的 值 赋 给 第 三 个 变量 。 这 样 
一 来 ， 第 三 个 变量 也 就 成 了 浮 点 型 的 变量 了 。 

但 是 话 又 说 回来 了 ，PHP 处 理 引 擎 也 不 是 回回 都 能 认 准 变量 的 数据 类 型 。 在 这 种 情况 
下 ， 就 得 告诉 PHP 引擎 想 存储 什么 类 型 的 数据 ， 而 不 是 让 PHP 处 理 引擎 自己 去 猜测 。 在 
脚本 中 可 以 用 如 下 的 语句 来 为 变量 指定 数据 类 型 。 

【 例 5.2】 为 变量 指定 数据 类 型 。 


1 S$newInt = (int) Svarl; // 指 定 变量 $Svarl 的 数据 类 型 为 整 型 
2 S$newFloat = (float) $varl; // 指 定 变 量 $varl 的 数据 类 型 为 浮 点 型 
3 S$newString = (string) S$varl; // 指 定 变量 $varl 的 数据 类 型 为 字符 串 型 


通过 使 用 类 似 例 5.2 中 的 语句 ， 就 可 以 将 等 号 右边 的 值 按照 括号 中 指定 的 数据 类 型 存 
储 到 等 号 左边 的 变量 中 。 可 是 ， 即 使 是 使 用 指定 数据 类 型 也 不 见得 就 万 无 一 失 。 要 是 当 了 
乔 太守 ， 乱 点 了 名 鸣 谱 ， 出 了 错 连 找 都 找 不 着 哪儿 错 了 。 因 为 PHP 处 理 引 擎 不 会 对 你 的 错 
误 报错 的 。 比 如 下 面 的 示例 。 

【 例 $.3】 为 变量 指定 错误 的 数据 类 型 。 

1 $varl I:05 


2 $var2 2 

3 S$newVarl = (int) $varl; 
4 

再 


$suml S$varl + S$var2; 
$sum2 $newVarl + $var2; 


在 如 例 5.3 所 示 的 脚本 中 ， 读 者 觉得 变量 gsuml 和 $sum2 的 值 会 是 多 少 呢 ? 是 2.8 呢 ， 
3 呢 ， 还 是 4 呢 ? 

先 来 分 析 一 下 这 五 个 变量 的 数据 类 型 : $varl 为 浮 点 型 变量 ，$var2 为 整 型 变量 ， 
SnewVarl 为 整 型 变量 。 按 照 PHP 处 理 引 擎 的 习惯 , Ssuml 应 该 是 个 浮 点 型 的 变量 , 而 $sum2 
则 是 个 整 型 变量 。 
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再 来 分 析 一 下 变量 $suml 和 $sum2 的 值 : 按说 $suml 的 值 肯 定 是 2.8， 这 个 没有 问题 。 
关键 是 如 果 $newVarl 的 值 如 果 还 是 1.8 的 话 ， 那 就 不 可 能 是 个 整 型 数据 了 ， 如 果 $newVarl 
是 个 整 型 数据 的 话 ， 那 它 的 值 就 肯定 不 是 1.8。 如 果 在 第 三 行 后 输出 SnewVarl 的 值 ， 你 会 
惊讶 的 发 现 , SnewVarl 的 值 居 然 是 1。 也 就 是 说 PHP 处 理 引 擎 对 变量 $varl 做 了 取 整 处 理 。 
这 样 一 来 ，$sum2 的 值 就 成 了 2。 

想 一 想 , 如果 在 脚本 中 随意 指定 了 变量 的 数据 类 型 会 发 生 什 么 问题 吧 。 几 千 行 的 代码 ， 
系统 也 没有 报错 ， 那 就 只 能 一 行 一 行 的 找 了 。 所 以 说 ， 不 到 万 不 得 已 ， 千 万 不 要 自行 指定 
数据 类 型 ， 一 旦 指定 错误 ， 恐 怕 你 真得 去 面壁 了 。 


5.2” 玩 转 数 字 一 一 整 型 和 浮 点 型 数据 


按照 之 前 的 说 法 ， 可 以 通过 为 变量 赋值 来 定义 变量 。 给 变量 赋 什 么 类 型 的 值 ， 变 量 就 
是 什么 类 型 的 变量 。 无 论 是 整 型 数据 ， 还 是 浮 点 型 数据 ， 都 是 数字 类 型 的 数据 。 它 们 可 以 
通过 以 下 语句 赋值 ，PHP 处 理 引擎 会 自动 识别 其 数据 类 型 。 

【 例 5.4】 定义 整 型 和 浮 点 型 变量 。 


1 S$intVar = 1; 
2 $floatVar = 2.6; 


5.2.1 四 则 运算 


在 PHP 中 , 可 以 用 数学 运算 符 连接 两 个 数字 或 者 数字 类 型 的 变量 来 对 数字 类 型 的 数据 
进行 四 则 运算 。 比 如 ， 使 用 加 号 (+) 连接 两 个 数字 ，PHP 处 理 引 擎 会 给 出 这 两 个 数字 相 
加 的 结果 ， 如 : 

亲生 全 和 


也 可 以 用 运算 符 连 接 两 个 变量 ， 如 : 
Svarl + $var2; 


需要 说 明 的 是 ， 如 果 需 要 对 变量 的 值 进行 四 则 运算 ， 千 万 不 要 在 定义 变量 的 时 候 用 引 
号 把 变量 的 值 包 于 起 来 。 否则 PHP 处 理 引擎 会 将 其 自动 识别 为 字符 串 型 的 数据 ,进而 在 进 
行 某 些 四 则 运算 的 时 候 出 错 。 来 看 下 面 一 段 脚本 。 

【 例 5.5】 字符 串 与 数字 相 加 的 结果 (一 ) 。 


$intVarOne = "1"; 

$intVarTwo = 2; 

$totalA = $intVarOne + $intVarTwo; 

echo "The value of total A is ".$totalA.".<br>"; 


$varOne = "x"; 
StotalB = $varOne + $intVarTwo; 
echo "The value of total B is ".$totalB."."; 


那么 PHP 会 怎么 处 理 这 段 脚本 呢 ? 我 们 知道 PHP 处 理 引擎 在 运算 时 会 根据 变量 的 实 
际 情况 来 自动 转换 变量 的 数据 类 型 ， 那 么 在 计算 变量 StotalA 的 值 时 ，PHP 是 会 将 变量 


yh 


co wwm 必 mwNP 
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$intVarOne 的 值 转换 成 一 个 整 型 数据 呢 ， 还 是 会 将 变量 SintVarTwo 的 值 转换 成 一 个 字符 串 
呢 ? 如 果 是 把 变量 $intVarOne 的 值 转换 成 一 个 整 型 数据 的 话 , 那 在 计算 变量 $totalB 的 时 候 ， 
变量 $varOne 的 值 又 会 转换 成 什么 呢 ? 在 回答 这 些 问题 之 前 ， 我 们 先 来 看 看 结果 ， 如 图 5-1 
所 示 。 


ee I 
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The value of total A is 3. 
The value of total B is 2. 


图 5-1 字符 串 和 数字 相 加 的 结果 


我 们 发 现 变量 $totalA 的 值 为 3， 也 就 是 说 PHP 处 理 引 擎 在 计算 变量 $totalA 的 值 时 ， 
自动 将 变量 $intVarOne 的 值 由 字符 串 型 数据 转换 成 了 整 型 数据 。 但 是 有 些 看 不 懂 的 是 ， 为 
什么 变量 $totalB 的 值 是 2 呢 ? 难道 说 PHP 处 理 引 擎 自动 将 变量 $varOne 的 值 判定 为 0 了 不 
成 ? 

的 确 是 这 样 。 因 为 变量 $varOne 的 值 不 是 PHP 处 理 引 擎 想 转换 就 能 转换 得 了 的 。 对 于 
这 种 无 法 转换 但 是 又 必须 代入 计算 的 时 候 ，PHP 处 理 引 擎 就 将 其 值 转换 为 整 型 数据 0， 然 
后 再 代入 计算 。 
虽然 说 PHP 处 理 引擎 自动 转换 数据 类 型 的 功能 很 强大 , 但 是 千 万 不 要 认为 它 回回 都 能 
转换 出 你 想 要 的 数据 类 型 。 再 来 看 一 段 脚本 。 

【 例 $.6】 字符 串 与 数字 相 加 的 结果 (二 ) 。 


本 S$intVarone "2 O00ns 

2 S$intVarTwo = 12; 

3 S$totalA = $intVarOne + $intVarTwo; 

4 echo "The value of total A is ".$totalA.".<br>"; 


大 家 可 以 先 猜测 一 下 : 这 段 脚 本 的 运行 结果 到 底 是 不 是 2012 呢 ? 

我 既然 这 么 不 怀 好 意 地 问 大 家 ， 答 案 自然 是 否定 的 。 这 段 脚 本 的 运行 结果 ， 也 就 是 变 
量 $totalA 的 值 ， 其 实 是 14。 为 什么 呢 ? 

因为 PHP 处 理 引 擎 不 认识 数字 断 位 符 ， 它 在 进行 数据 类 型 转换 时 看 到 逗号 就 停 下 来 
了 ， 于 是 $intVarOne 的 值 被 自动 转换 成 了 整 型 类 型 的 2。 这 样 代 入 计算 的 话 ， 变 量 totalA 
的 值 自然 是 14， 而 不 是 2012 了 。 

当然 ，PHP 支持 的 数学 运算 符 不 是 只 有 加 号 (+) ， 表 5-1 列 出 了 PHP 支持 的 五 种 数 

表 5-1 PHP 支 持 的 数学 运算 符 
运算 符 描述 

十 | 取 两 数 或 两 变量 之 和 ， 如 “echo $varl + $var2;” 或 “echo 1+2;”， 其 值 为 3 
取 两 数 或 两 变量 之 差 ， 如 “echo $varl - $var2;” 或 “echo 3 - 1;”， 其 值 为 2 
取 两 数 或 两 变量 之 积 ， 如 “echo $varl * $var2;” 或 “echo 2 * 2;”， 其 值 为 4 


描述 


/ | 取 两 数 或 两 变量 之 商 ， 如 “echo $varl/$var2;” 或 “echo 4/2;”， 其 值 为 2 


% 取 两 数 或 两 变量 之 余 ， 如 “echo $varl1%$var2;” 或 “echo 5%3:;”， 其 值 为 2 


表 5-1 所 示 中 列 出 的 五 种 运算 符 中 ， 第 五 种 是 用 来 求 两 数 的 余数 。 前 四 种 就 相当 于 数 
学 里 的 加 减 乘除 。 在 使 用 它们 进行 四 则 运算 时 也 遵守 加 减 乘除 的 法 则 ， 即 按照 先 乘除 后 加 
减 的 顺序 从 左 往 右 依次 计算 。 在 书写 运算 式 时 , 也 可 以 使 用 括号 , 括号 里 的 内 容 优 先 计算 。 
这 些 都 是 我 们 在 学 小 学 数学 时 都 已 经 掌握 了 的 。 下 面 来 看 儿 个 例子 。 


【 例 $5.7】 四 则 运算 。 


1 S$var 
2 S$var 
3 $var 
4 $var 


= (1 + 2) *3+1; // 第 一 步 ， 先 计算 括号 内 的 加 法 
33 // 第 二 步 ， 计 算 乘法 
S94 纪 // 第 三 步 ， 计 算 加 法 


Sa 


对 于 像 “$var = $var + 1” 这 样 的 句子 ， 可 以 简写 成 Svart++。 类 似 的 ， 像 “S$var = $var - 
1” 这 样 的 句子 ， 可 以 简写 成 $var --。 
【 例 $.8】 PHP 脚本 中 的 缩 略 名 (一 》。 


hh $var = 0; 

2 SVar++7 

3 echo $var; // 此 时 输出 的 变量 $var 的 值 为 1， 相 当 于 echo $var + 1; 

4 $var-——; 

5 echo $var // 此 时 输出 的 变量 $var 的 值 为 0， 相 当 于 echo $var - 17 

对 于 在 某 变量 自身 上 或 加 、 或 减 、 或 乘 、 或 除 一 个 常数 的 情况 ，PHP 脚本 都 提供 了 缩 


略 铅 。 比 如 例 5.9。 
【 例 5.9】 PHP 脚本 中 的 缩 略 句 (二 ) 。 


$var 
echo 
$var 
echo 
$var 
echo 
$var 
echo 


OooODNp 


$var = 10; 


忧 二 之 和 


$var; 


= 
$var 
*#=2; 


Svar7 


/=3; 


Svar7 


5.2.2 复杂 运算 


// 此 时 输出 的 变量 $var 的 值 为 12， 相 当 于 echo $var +2; 
// 此 时 输出 的 变量 $var 的 值 为 9， 相当 于 echo $var - 3; 
// 此 时 输出 的 变量 $var 的 值 为 18， 相 当 于 echo $var * 2; 
// 此 时 输出 的 变量 $var 的 值 为 56， 相当 于 echo $var / 3; 


这 里 说 的 复杂 运算 其 实 也 复杂 不 到 哪儿 去 。PHP 提供 了 很 多 的 函数 用 于 诸如 求 平方 根 
之 类 的 运算 。 这 样 一 来 ， 就 省 去 了 写 脚本 的 麻烦 。 只 用 写 上 一 句 简单 的 语句 就 能 实现 这 类 
运算 。 关 于 函数 的 内 容 ， 我 们 会 在 第 9 章 中 讲 到 。 现 在 来 看 一 个 例子 。 


【 例 $.10】 求 平方 根 。 


1 $varSqrt = sqrt(81); 


2 echo $varSqrt; 


页 去 
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在 例 5.10 中 ， 我 们 使 用 sqrt0 函 数 计算 了 81 的 平方 根 ， 并 将 计算 结果 赋值 给 了 变量 
$varSqrt， 然 后 使 用 echo 语句 输出 了 该 变量 的 值 。 在 这 里 只 用 了 一 个 简单 的 函数 就 完成 了 
求 平方 根 的 任务 。 

在 PHP 语言 中 ， 像 这 样 的 函数 有 很 多 ， 大 多 数 都 能 在 平常 编写 脚本 时 用 到 ， 节 省 了 大 
量 的 编码 时 间 。 再 来 看 两 个 示例 。 

【 例 S.11】 上 取 整 和 下 取 整 。 


1 S$varCeiling = ceiling (6.52); 

[4 SvarFloor = floor(6.52); 

党 $varTotal = $varCeling +$varFloor; 

4 echo $varTotal; 

在 PHP 中 ，ceiling0 和 floor0 两 个 函数 都 是 用 来 对 浮 点 型 数 取 整 用 的 。 前 者 为 上 取 整 
函数 ， 后 者 为 下 取 整 函数 。 所 谓 上 取 整 是 指 将 某 浮 点 数 的 整数 部 分 加 1 作为 取 整 后 的 值 ， 
而 取 值 是 指 保留 该 浮 点 数 的 整数 部 分 作为 取 整 后 的 值 。 

在 例 5.11 中 ， 我 们 使 用 这 两 个 函数 对 6.52 这 个 浮 点 数 进 行 了 取 整 。 按 照 上 取 整 和 下 
取 整 的 定义 ， 可 知 $varCeiling 的 值 为 7， 而 $varFloor 的 值 为 6。 脚本 的 第 三 行 定义 了 变量 
$varTotal， 并 为 其 赋值 SvarCeiling 和 $varFloor 之 和 。 因 此 ， 脚 本 第 四 行 输出 的 $varTotal 的 
值 应 该 为 13。 大 家 可 以 自行 验证 一 下 。 


5.2.3 ”数字 格式 化 


在 前 面 两 节 里 , 我 们 使 用 echo 命令 输出 的 数字 都 很 小 。 对 于 比较 长 的 数字 ,我 们 一 时 
半 会 就 很 难 认 出 它 的 大 小 来 。 为 了 解决 这 个 问题 ， 人 们 便 以 每 三 位 加 一 个 逗号 的 形式 把 数 
字 给 隔 开 。 这 样 隔 开 以 后 ， 再 长 的 数字 也 变 得 易 读 起 来 。 

在 编写 PHP 脚本 时 ， 也 可 以 让 系统 输出 带 这 种 逗号 的 数字 来 。 方 法 也 很 简单 ， 只 要 使 
用 number_format0 函 数 就 好 了 。 不 过 这 个 number_format(0) 函 数 的 参数 比较 多 。 具 体格 式 
如 下 : 


number format (number, decimals, "decimalsep","thousandsep") 


这 四 个 参数 中 只 有 第 一 个 参数 是 必 选 的 ， 其 他 参数 可 选 。 那 么 它们 都 代表 着 什么 意思 
呢 ， 我 们 一 个 一 个 来 看 。 

口 number: 代表 的 就 是 需要 格式 化 的 数字 ， 这 个 参数 是 必 选 的 。 

口 decimals: 指 的 是 小 数位 数 ， 这 个 参数 是 可 选 的 。 如 果 没 有 指定 小 数位 数 ， 则 该 数 
字 就 会 被 四 舍 五 入 为 一 个 整数 后 输出 ,当然 ,如果 指 定 了 thousandsep, 那么 decimals 
就 是 必 选 的 了 。 

口 decimalsep: 指 的 就 是 用 于 分 隔 整 数 部 分 和 小 数 部 分 的 符号 ， 这 个 参数 是 可 选 的 。 
默认 使 用 的 符号 为 点 〈.) 。 当 然 ， 如 果 指 定 了 decimalsep， 那 么 thousandsep 也 就 
成 了 必 选 参数 了 。 

口 thousandsep: 指 的 就 是 用 于 分 隔 百 位 与 千 位 的 符号 ， 这 个 参数 也 是 可 选 的 。 默 认 
使 用 的 符号 为 逗号 〈(,) 。 当 然 ， 如 果 指 定 了 thousandsep， 那 么 decimalsep 也 就 成 
了 必 选 参数 了 。 

现在 来 看 一 些 例子 。 由 于 该 函数 的 使 用 比较 简单 ， 下 面 就 以 表格 的 形式 为 大 家 展示 一 


。T4 。 
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下 该 函数 的 使 用 方法 和 可 能 出 现 的 结果 ， 如 表 5-2 所 示 。 


表 5-2 number_format() 函 数 的 参数 使 用 及 其 输出 
Snumber 输出 结果 


12321 number format($number) 12,321 
12321.66 number format($number, 2) 12,321.66 
12321.66 number format($number) 12,322 
12321.6 number format($number, 3) 12,321.600 
12321 number format($number, 0,".", ".") 12.321 


12321.66 number format($number, 2,".", "") 12321.66 
S$.3 咬文嚼字 一 字符 串 型 数据 


按照 之 前 的 讲法 ， 字 符 串 型 数据 可 以 用 来 存储 字符 信息 。 那 么 ， 什 么 是 字符 呢 ? 所 谓 
字符 ， 指 的 就 是 字母 、 数 字 和 标点 符号 等 一 切 可 以 打印 在 屏幕 上 的 符号 。 那 么 字符 串 指 的 
自然 就 是 一 连 串 的 字符 了 。 如 果 一 个 数字 被 存储 为 字符 串 型 的 数据 时 ， 它 仅仅 就 是 一 个 字 
符 或 字符 串 ， 不 能 再 用 于 数学 运算 。 比 如 ， 我 们 经 常会 把 电话 号 码 和 身份 证 号 码 之 类 的 信 
息 保 存 为 字符 串 型 的 数据 ， 虽 然 它们 大 多 数 都 是 纯粹 由 数字 构成 的 。 我 们 可 以 通过 以 下 的 
语句 把 变量 定义 成 字符 串 类 型 的 变量 。 

【 例 $.12】 定义 字符 串 型 变量 。 
时 $phoneNo = "15946892256"; 
$authorizationCode = "A34279"; 


2 
$activationCode = (string) $phoneNo.$authorizationCode; 
4 echo "Your activation code is ".$activationCode."."; 


在 例 5.12 中 , 定义 了 一 个 变量 用 于 存放 电话 号 码 , 还 定义 了 一 个 变量 用 于 存放 授权 码 ， 
然后 把 两 个 变量 连接 在 一 起 组 成 激活 码 ， 最 后 把 激活 码 打印 在 屏幕 上 。 最 后 运行 的 结果 应 
该 是 如 下 的 一 句 话 : 

You activation code is 15946892256A34279. 


5.3.1 文字 游戏 


在 本 小 节 里 ， 我 会 传授 给 大 家 几 招 比较 常见 且 十 分 有 用 的 用 于 处 理 字 符 串 的 招式 。 大 
家 在 处 理 字 符 串 时 可 以 用 到 它们 。 

第 一 招 倒 背 如 流 

在 处 理 字符 串 时 ， 有 时 会 要 求 把 一 个 字符 串 倒 过 来 写 。 其 实 这 是 一 种 十 分 简单 的 加 密 
方式 。PHP 为 我 们 提供 了 一 个 预 置 函 数 strrevO。 使 用 这 个 函数 ， 就 可 以 把 任何 字符 串 反 转 

【 例 $.13】 字符 串 反 转 函数 。 


[ SvarString = "This is a fine day!™"; 
2 $varReverse = strrev(SvarString) 7 
3 echo $varReverse; 


。7T5。 
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执行 例 5.13 中 的 脚本 ， 会 得 到 什么 结果 呢 ? 


CE | 


7 了 加 出 7 pagev Safeyv Toosv 右 - 


!yad enif a si sihT 


图 5-2 字符 串 反 转 函 数 


结果 就 是 如 图 5-2 所 示 ， 出 现 了 一 段 看 上 去 像 是 乱码 的 文字 。 再 仔细 辨认 一 下 ， 才 发 
现 , 原来 这 个 函数 会 把 字符 串 的 最 后 的 一 个 字符 放 到 最 前 面 , 倒数 第 二 个 字符 放 到 第 二 位 ， 
以 此 类 推 ， 实 现 反 转 的 目的 。 这 样 倒 背 如 流 的 功夫 真是 让 人 羡慕 啊 。 

第 二 招 ”伸缩 自如 

有 时 候 ， 我 们 希望 把 一 个 字符 串 中 的 英文 字母 全 部 变 成 大 写 的 ， 有 时 候 又 希望 把 它们 
全 部 变 成 小 写 的 ， 有 时 候 我 们 希望 让 每 个 单词 的 首 字母 大 写 ， 有 时 候 又 希望 让 句子 的 第 一 
个 字母 大 写 。 我 们 的 要 求 真是 多 的 让 人 难以 满足 啊 。 不 过 PHP 还 是 提供 了 一 批 函数 来 帮助 
我 们 达成 愿望 ， 让 字符 串 们 像 孙 悟空 的 金 夭 棒 一 样 可 以 伸缩 自如 、 忽 大 忽 小 啊 。 

【 例 5.14】 字符 串 大 小 写 函数 。 
$varSur = "windsor"; 
SvarGiven = "william"; 
S$varAffixTitle = "prince"; 
SvarSuffixTitle = "duke of cambridge"; 


echo ucfirst ($varAffixTitle)." ".ucfirst ($varGiven)." ".strtoupper 
($varSur) .", ".ucwords ($varSuffixTitle)."<br>"; 


PRODP 


这 段 脚本 的 执行 结果 如 图 5-3 所 示 。 
-- mm | 


jx 
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| Prince William WINDSOR, Duke Of Cambridge 


图 5-3 字符 串 大 小 写 函数 


在 例 5.14 所 示 的 脚本 中 ， 定 义 了 三 个 函数 ， 然 后 分 别 为 它们 赋值 。 大 家 可 以 注意 到 ， 
用 来 赋值 的 字符 串 中 所 有 的 英文 字母 都 是 小 写 的 。 在 输出 的 时 候 ， 使 用 了 三 个 函数 ， 它 们 
分 别 是 ucfirst)、ucwords0 和 strtoupper()。 现 在 对 照 脚 本 的 执行 结果 来 看 看 这 三 个 函数 的 
功能 。 

在 上 述 脚 本 里 ， 


。76 。 
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口 函数 ucfirst0 中 使 用 两 次 ,代入 的 变量 分 别 是 SvarAffixTitle 和 $varGiven， 显示 的 结 
果 为 首 字母 大 写 。 

口 函数 ucwords0 使 用 了 一 次 ， 代 入 的 变量 为 SvarSuffixTitle， 显 示 的 结果 为 字符 串 的 
每 个 单词 首 字母 大 写 。 

口 函数 strtoupper0 使 用 了 一 次 ， 代 入 的 变量 为 SvarSur， 显 示 的 结果 为 字符 串 每 个 字 
母 都 大 写 。 

其 实 ，PHP 还 提供 了 一 个 函数 用 来 把 字符 串 中 的 所 有 英文 字母 都 变 成 小 写 的 ， 这 个 函 


数 就 是 strtolower()。 

这 些 函 数 名 称 看 上 去 似乎 没有 什么 规律 ， 应 该 怎么 记 呢 ? 如 果 你 懂 点 英文 ， 其 实 很 
好 记 : 

口 ucfirst 就 是 upper case first 的 缩写 ,意思 是 字符 串 首 字母 大 写 ; 

口 ucwords 就 是 upper case words 的 缩写 ,注意 words 是 复数 形式 ， 意 思 是 字符 串 每 

个 单词 的 首 字母 大 写 ; 

口 strtoupper 就 是 string to upper 的 缩写 ， 意 思 是 字符 串 所 有 的 字母 都 大 写 ; 

口 strtolower 就 是 string to lower 的 缩写 ， 意 思 是 字符 串 所 有 的 字母 都 小 写 。 

第 三 招 断章取义 
虽然 我 们 平素 里 对 断章取义 这 种 行为 深恶痛绝 ， 但 该 断章取义 的 时 候 ， 我 们 还 得 断 章 
取 义 不 是 ? PHP 提供 了 这 么 一 个 函数 ， 它 就 是 substr0。 它 有 三 个 参数 ， 分 别 是 字符 串 、 起 
始 位 置 和 子 串 长 度 。 子 串 长 度 可 以 不 指定 ， 这 样 的 话 ， 截 取 的 子 串 包 含 的 就 是 从 指定 位 置 
到 字符 串 结尾 的 所 有 字符 了 。 我 们 还 是 先 来 看 一 段 脚本 。 

【 例 $.1S】 字符 串 截 取 函 数 (一 ) 。 


1 S$varSstring = "Hello, this is my World!"; 

交 $firstSub = substr($varstring, 0, 5); 

3 $secondSub = substr($varstring, 18); 

4 echo $firstSub." ".$secondsub; 

在 例 5.15 这 段 脚本 中 , 定义 了 变量 $varString， 并 为 其 赋值 “Hello, this is my World!”。 
然后 ， 再 使 用 substr0 函 数 截取 了 变量 $varString 的 部 分 内 容 定义 了 两 个 变量 $firstSub 和 
$secondSub， 在 这 里 强调 一 下 ，substr0 函 数 三 个 参数 分 别 代表 变量 名 、 起 始 位 置 和 子 串 长 
度 。 还 要 强调 一 下 ， 函 数 substr0 是 从 第 0 位 开始 记录 字符 串 中 每 个 字符 的 位 置 的 ， 也 就 是 
说 字符 串 的 第 一 个 字符 所 在 位 置 为 0, 第 二 个 字符 所 在 位 置 才 是 1。 按照 这 个 说 明 ， 大 家 可 
以 自己 推演 一 下 脚本 的 运行 结果 。 

如 果 读 者 足够 心细 ， 数 数 没有 问题 的 话 ， 应 该 会 得 到 如 图 5-4 所 示 的 结果 。 


www greatwallteaychapte05/E5-15php |B | | x sg 
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Hello World! 


图 5-4 字符 串 截取 函数 


Ts 


第 2 篇 常量 、 变 量 与 数据 


有 的 同学 可 能 会 说 了 ， 我 还 得 靠 数 数 才能 确定 某 个 字符 所 在 的 位 置 ， 如 果 字 符 串 很 长 
的 话 ， 那 岂 不 是 数 数 都 要 数 叶 血 了 ? 

其 实 没有 那么 夸张 ， 如 果实 在 不 想 这 么 复杂 的 话 ， 还 有 变通 的 方法 。 这 里 我 们 把 如 例 
5.15 所 示 脚 本 的 第 三 行 修改 成 如 下 的 样子 ， 运 行 的 结果 还 是 一 样 的 。 

3 S$secondSub = substr($varstring, -6); 


上 面 这 一 句 脚本 是 从 字符 串 的 末端 开始 数 数 ， 每 个 字符 串 最 后 一 个 字符 的 位 置 为 第 -1 
位 。 我 们 需要 截取 “World!” 这 个 子 串 ， 数 数 看 ， 应 该 是 从 第 -6 位 起 到 字符 串 末 尾 止 。 

在 这 里 ， 顺 便 再 说 一 下 ， 如 果子 串 长 度 为 负 N CN 为 整数 ) 的 话 ， 表 明 截 取 的 子 串 包 
含 从 指定 位 置 起 到 字符 串 末 尾 倒数 第 N+1 位 的 所 有 字符 。 青 来 看 一 段 脚本 : 

【 例 5.16】 字符 串 截 取 函 数 (二 ) 。 


和 SvarString = "Hello, this is my World!"; 
2 $firstSub = substr ($varstring, 4, 1); 

3 $secondSub = substr($varString, -9, -1); 
4 echo $firstSub.", ".$secondSub; 


这 段 脚 本 的 结果 应 该 是 : 
o, my World 
你 数 对 了 吗 ? 


5.3.2 文本 格式 化 


在 5.2.3 小 节 里 ， 我 们 知道 了 如 何 格 式 化 数字 。 在 本 小 节 里 ， 我 们 将 要 学 习 如 何 格 式 
化 文本 ， 也 就 是 让 文本 以 我 们 指定 的 格式 输出 。 对 于 文本 来 说 ， 定 义 输出 格式 是 一 件 非常 
复杂 的 事情 ， 就 像 在 Office 的 Word 组 件 中 输出 文字 是 十 分 容易 的 事情 ， 但 是 让 一 个 不 太 
懂 如 何 使 用 Word 工具 栏 对 输出 的 文本 进行 格式 的 排版 却 通常 比 写 这 些 文字 耗费 的 时 间 还 
要 多 。 

中 国人 有 一 句 话说 的 好 : 难 者 不 会 ， 会 者 不 难 。 只 要 掌握 了 技巧 ， 定 义 好 文本 的 输出 
格式 将 是 一 件 十 分 容易 的 事情 。 

在 本 小 节 里 ， 将 介绍 两 个 以 指定 格式 输出 指定 变量 的 函数 ， 它 们 分 别 是 printtD 和 
sprintf()。 前 者 相当 于 echo， 可 以 直接 输出 内 容 ， 后 者 则 像 number_format()， 只 不 过 功能 
更 加 强大 。 它 不 光 可 以 对 文本 进行 格式 化 输出 ， 对 于 数字 也 一 样 操控 自如 。 这 两 个 函数 的 
参数 如 下 : 


printf ("format", $varNamel, $varName2, ...); 
SnewVar = sprint("format", $varNamel, $varName2, ..); 


来 看 下 面 一 段 脚本 。 
【 例 $.17】 格式 化 文本 (一 )。 
Snboys = 3; 


区 Sngirls = 5; 
3 printf("%s boys and %s girls", $nboys, $ngirls); 


运行 这 段 脚本 后 ， 显 示 的 结果 如 图 5-5 所 示 。 
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图 5-5 函数 printf0 的 输出 结果 


在 这 段 脚本 中 ， 有 一 个 特殊 的 符号 “%s”， 这 就 是 定义 文本 输出 格式 的 符号 。 就 是 这 
个 “%s” 告 诉 PHP 处 理 引 擎 将 后 续 指定 的 变量 的 值 作 为 字符 串 按 顺序 插入 到 指定 的 位 置 。 
其 中 “%” 告 诉 PHP 处 理 引擎 从 这 里 开始 执行 文本 格式 化 操作 。 其 实 这 个 指令 的 真实 模样 
如 图 5-6 所 示 。 


pad width dpe 
和 rn yp 


图 5-6 文本 格式 化 指令 


按照 图 5-6 所 示 ， 这 个 指令 由 六 个 部 分 组 成 。 下 面 分 别 对 这 六 个 部 分 进行 说 明 


口 
口 


口 


口 


% 《起 始 符 ) ， 用 于 告诉 PHP 处 理 引擎 从 这 里 开始 执行 文本 格式 化 操作 。 

pad 〈 占 位 符 ) ， 用 于 在 输出 的 文本 长 度 小 于 指定 长 度 时 补 齐 两 者 之 间 的 差距 时 所 
使 用 的 占 位 符 。 比 如 某 待 输出 变量 的 值 的 长 度 为 4， 而 指定 长 度 为 8， 那么 输出 的 
文本 长 度 为 8， 但 其 中 4 位 为 占 位 符 。 该 占 位 符 可 以 是 0、 空格 或 任意 字符 。 但 需 
要 注意 的 是 ， 如 果 该 占 位 符 为 字符 ， 那 么 它 的 前 面 应 该 有 一 个 单 引号 (7) 。 

-〔 对 齐 符 ) ， 用 于 告诉 PHP 处 理 引擎 如 何 处 理 待 输 出 某 变量 的 值 与 占 位 符 的 左右 
关系 。 若 使 用 了 对 齐 符 ， 待 输出 变量 左 对齐 ， 即 待 输出 变量 在 占 位 符 的 左边 。 若 
省 略 了 对 齐 符 ， 则 待 输出 变量 右 对 齐 ， 即 待 输出 变量 在 占 位 符 的 右边 。 还 是 用 之 
前 的 例子 ， 某 待 输出 变量 的 值 长 度 为 4 而 指定 的 长 度 为 8， 那么 输出 的 文本 长 度 为 
8， 但 其 中 4 位 为 占 位 符 。 若 使 用 了 对 齐 符 ， 则 待 输出 变量 的 值 在 4 位 占 位 符 左 边 
输出 ， 若 未 使 用 对 齐 符 ， 则 待 输出 变量 的 值 在 4 位 占 位 符 右边 输出 。 

width (指定 长 度 ) ， 用 于 告诉 PHP 处 理 引擎 输出 的 字符 串 长 度 。 若 待 输出 变量 的 
值 的 长 度 与 该 处 指定 的 长 度 有 差距 ， 则 使 用 之 前 指定 的 占 位 符 来 补 齐 。 比 如 ， 待 
输出 变量 的 值 为 1， 指 定 长 度 为 5， 占 位 符 为 0， 未 使 用 对 齐 符 ， 则 输出 的 文本 为 
00001。 

-dec《〈 小 数位 数 ) ， 用 于 告诉 PHP 处 理 引 擎 在 输出 整形 变量 或 浮 点 型 变量 时 需要 
保留 的 小 数位 数 。 一 定 要 注意 保留 前 面 的 点 (.〉。 

type《〈 变 量 类 型 ) ， 有 两 种 类 型 可 选 : 字符 串 型 ， 用 “s” 表 示 ; 浮 点 型 ， 用 “f” 
表示 。 


那 在 例 5.17 中 ， 我 们 看 到 的 “%s” 其 实 就 是 告诉 PHP 处 理 引 擎 以 字符 串 的 形式 输出 
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指定 变量 的 值 。 为 了 巩固 上 面 讲 到 的 知识 ， 我 们 再 来 看 一 段 脚本 。 
【 例 5.18】 格式 化 文本 (二) 。 


业 Scostl1 = 456; 

2 $commodityl = "Microwave Oven"; 

ei Secost2 = 3295 

4 $commodity2 = "Men's Underware"; 

2 $todayDate = "Jan. 21st, 2012"; 

6 S$currentTime = "19:38:46"; 

统 $timeStamp = sprintf ("Welcome to Goodbuy <br> %-25s%s <br>", $todayDate, 
$currentTime); 

8 $billEntryl = sprintf("%$'.-25s%5.2f <br>", $commodityl, $cost1); 

9 $billEntry2 = sprintf("%'.-25s%5.2f <br>", $commodity2, $cost2); 


10 echo $timeStamp; 
11 echo $billEntryl; 
12 echo $billEntry2; 


这 段 脚 本 输出 的 内 容 如 图 5-7 所 示 。 


Ex 
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图 5-7 文本 格式 化 (二) 


运行 例 5.18 中 的 脚本 ， 我 们 打印 了 一 张 超市 的 小 票 。 在 脚本 中 ， 使 用 sprintf0 函 数 定 
了 三 个 段落 。 第 一 个 段落 为 超市 欢迎 辞 和 小 票 打 印 的 日 期 加 时 间 ， 第 二 个 段落 为 第 一 件 

商品 的 名 称 和 价格 ， 第 三 个 段落 为 第 二 件 商品 的 名 称 和 价格 。 

在 这 三 次 使 用 sprinf0) 函 数 的 过 程 中 ， 我 们 向 PHP 处 理 引擎 下 达 了 四 个 格式 指令 ， 它 
们 分 别 是 “%-25s” 、“ %s”、“ %'.-25s” 和 “%5.2f”。 第 一 个 指令 要 求 输出 一 和 
为 25 的 字符 串 , 该 字符 串 包含 以 左 对 齐 方式 输出 的 指定 变量 的 值 和 作为 占 位 符 的 空格 。 第 
二 个 指令 要 求 将 指定 变量 的 值 以 字符 串 的 形式 输出 。 第 三 个 指令 要 求 输出 一 串 长 度 为 25 
的 字符 串 ， 该 字符 串 包 含 以 左 对 齐 方式 输出 的 指定 变量 的 值 和 作为 占 位 符 的 点 〈.) 。 第 四 
个 指令 则 要 求 将 指定 变量 的 值 以 字符 串 的 形式 输出 ,字符 串 的 长 度 为 5， 小 数 后 保留 2 位 。 

通过 图 5-7 所 示 的 内 容 ， 可 以 发 现 ， 网 页 上 两 条 消费 记录 的 前 后 都 是 对 齐 的 ， 而 时 间 
日 期 那 一 段 前 后 并 没有 和 下 面 的 两 条 消费 记录 对 齐 。 但 是 通过 观察 源 代码 ， Vi 
三 个 段落 的 长 度 都 是 一 样 的 , 均 为 25 个 字符 。 也 就 是 说 我 们 用 作 占 位 符 的 空格 被 浏览 器 
吃 掉 了 。 


5.4 操控 时 间 


时 间 型 数据 


时 间 和 日 期 在 脚本 里 可 以 发 挥 非常 大 的 作用 。 读 者 可 以 使 用 脚本 进行 基于 时 间 的 计 
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算 ， 尤 其 是 当 需 要 给 分 布 在 世界 各 地 的 用 户 提供 当地 的 时 间 信 息 时 ， 时 区 换算 成 了 最 基本 
的 要 求 。 在 计算 机 的 世界 里 ， 时 间 是 以 时 间 戳 的 形式 存储 的 。 也 就 是 说 ， 所 有 的 时 间 信 息 
都 是 以 秒 为 单位 进行 记录 的 。 为 了 将 时 间 戳 里 记录 的 信息 转换 成 对 人 友好 的 格式 ，PHP 提 
供 了 许多 函数 。 

在 本 节 里 ， 我 们 将 通过 了 解 这 些 函 数 的 使 用 ， 来 学 习 如 何 使 用 PHP 脚本 操控 时 间 。 


5.4.1 时 间 格 式 记 
每 个 国家 都 有 着 自己 约定 俗 成 的 时 间 格 式 。 大 家 如 果 使 用 的 是 Windows 的 操作 系统 ， 


可 以 在 “控制 面板 ”中 找到 “地 区 和 语言 ”图 标 。 双 击 后 ， 可 以 看 到 如 图 5-8 所 示 的 “地 
区 和 语言 ”对 话 框 。 


Fomats | Location | Keyboards and Languages | Administrative 


Format: 
[Chinese (Simplified, PRC) 
Change sorting method 
Date and time formats 
Short date: yyyy/M/d 
Long date: 2 年 M 月 dB 


Shortime Hmm 


Long time: Hmmess 


First day of week | 星期 日 

What does the notation mean? 
Examples 

Short date: 2012/12117 
Long date: 2012 年 12 月 17 日 
Shorttime 2056 
Longtime 20:56:34 


图 5-8 ”Windows 操作 系统 中 的 地 区 与 语言 对 话 框 


在 “格式 ”选项 卡 中 ， 可 以 通过 选择 “格式 ”下 拉 列 表 框 内 的 不 同 国家 和 地 区 来 查看 
各 国 、 各 地 区 在 书写 时 间 和 日 期 的 时 遵循 的 格式 。 

在 PHP 脚本 中 ， 经 常 使 用 的 与 时 间 和 日 期 有 关 的 函数 就 是 date0。 这 个 函数 可 以 将 存 
储 在 时 间 戳 里 的 时 间 信 息 按照 我 们 指定 的 格式 显示 出 来 。 这 个 函数 格式 如 下 : 

date ("format", $timestamp); 

先 来 看 一 段 脚本 。 

【 例 S.19】 使 用 dateO 函 数 。 


1 S$today = date("Y/m/d"); 
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echo "Today is ".$today."."; 


假设 今天 是 2012 年 12 月 17 日 ， 


突 
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Today is 2012/12/17. 


图 5-9 使 用 dateO 函 数 


在 例 5.19 中 ， 只 指定 了 date0 函 数 两 个 参数 中 的 一 个 ， 并 没有 指定 第 二 个 参数 。 如 果 
没有 指定 第 二 个 参数 ，PHP 处 理 引擎 会 直接 引用 系统 当前 时 间 。 现 在 我 们 来 重点 关注 一 下 


脚本 中 
来 区 分 


的 第 一 个 参数 “Y/m/d”。 为 什么 Y 大 写 , 而 m 和 d 都 小 写 了 呢 ? 其 实 这 是 PHP 用 
寺 间 格式 的 方法 。 比 如 “y-m-d” 输 出 的 是 12-12-17, 而 “M.d.Y ”输出 的 是 Dec.17.12。 


表 5-3 列 出 了 我 们 在 定义 时 间 格 式 时 经 常用 到 的 一 些 符号 。 


表 
意 
月 份 的 英文 单词 缩 略 语 
月 份 的 英文 单词 全 拼 


月 份 的 数字 形式 ， 十 月 前 的 月 份 含 


5-3 ”时 间 与 日 期 格式 符 
实 举例 
Dec 
December 
?前 置 的 0 02 


符号 

M 

F 

m 

n 月 份 的 数字 形式 ， 十 月 前 的 月 份 不 含 前 置 的 0 2 
d 日 期 的 数字 形式 ， 十 日 前 的 日 期 含 前 置 的 0 02 
日 期 的 数字 形式 ， 十 日 前 的 日 期 不 含 前 置 的 0 区 

I 星期 中 的 某 一 天 对 应 英文 单词 的 全 拼 Friday 
D -星期 中 的 某 一 天 对 应 英文 单词 的 缩写 Fri 
Ww -星期 中 的 某 一 天 对 应 的 数字 ， 从 0 (星期 天 ) 到 6 (星期 六 ) 5 

区 年 份 ， 四 位 数字 2012 
y 年 份 ， 两 位 数字 12 
g 小 时 ， 取 值 在 0 到 12 之 间 ， 十 位 以 下 无 前 置 的 0 2 
G 小 时 ， 取 值 在 0 到 24 之 间 ， 十 位 以 下 无 前 置 的 0 加 

h 小 时 ， 取 值 在 0 到 12 之 间 ， 十 位 以 下 有 前 置 的 0 02 
H 小 时 ， 取 值 在 0 到 24 之 间 ， 十 位 以 下 有 前 置 的 0 02 

i 分 钟 00 

S 秒 钟 00 

a 上 午 (am) 或 下 午 (pm) am/pm 
A 上 午 (AM) 或 下 午 (PM) AM/PM 
U Unix 系统 计 秒 1056244941 

5.4.2 ”时 间 型 变量 
在 例 5.19 中 ， 定 义 了 一 个 变量 ， 这 个 变量 是 时 间 型 变量 吗 ? 答案 自然 是 否定 的 ， 那 个 
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变量 是 个 字符 串 型 变量 。 如 果 需 要 定义 一 个 时 间 型 的 变量 ， 首 先 得 知道 何 为 时 间 型 变量 。 
为 此 ， 我 们 先 来 看 一 段 脚本 。 

【 例 $.20】 获取 时 间 戳 。 

1 S$today = time(); 

2 echo $today; 

在 这 一 章 的 开始 ， 提 过 时 间 戳 这 个 概念 。 那 么 什么 是 时 间 戳 呢 ? 时 间 戳 其 实 是 计算 机 
用 来 存储 时 间 的 方式 。 时 间 戳 里 存储 的 时 间 是 以 秒 为 单位 的 ， 记 录 的 是 指定 时 间 与 1970 
年 1 月 1 日 0 时 0 分 0 秒 钟 之 间 的 时 间 差 ,在 例 5.20 中 , 使 用 当前 的 时 间 定 义 了 变量 $today。 
然后 输出 这 个 变量 ， 得 到 的 应 该 会 是 如 图 5-10 中 的 一 段 数字 。 
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图 5-10 输出 时 间 戳 
当 定 义 了 一 个 时 间 型 变量 之 后 , 就 可 以 在 上 一 节 中 提 到 的 函数 date0 中 引用 了 。 在 PHP 
中 ， 还 可 以 通过 其 他 方式 ， 将 指定 时 间 以 时 间 戳 的 形式 存储 到 变量 中 。 
【 例 $.21】 存储 指定 时 间 到 变量 中 (一): strtotime()。 
1 $importantDate = strtotime ("today"); // 输 出 当前 时 间 的 时 间 戳 


$importantDate = strtotime ("tomorrow"); // 输 出 24 小 时 后 的 时 间 戳 
$importantDate = strtotime ("last saturday") ;// 输 出 上 周 六 当前 时 间 的 时 间 惟 


$importantDate = strtotime ("2 weeks ago"); // 输 出 两 周 前 当前 时 间 的 时 间 惟 


$importantDate = strtotime("3 months later"); 


// 输 出 三 个 月 后 当前 时 间 的 时 间 截 
yi $importantDate = strtotime ("next year gmt"); 
// 以 格林 威 治标 准时 输出 明年 今日 当前 时 间 的 时 间 戳 
8 $importantDate = strtotime ("tomorrow 4am") 7 


// 输 出 明天 凌晨 四 点 当前 时 间 的 时 间 改 


大 家 一 定 觉得 很 神奇 ， 引 号 里 不 就 是 一 些 英语 短语 么 ? 居然 可 以 这 么 用 。 其 实 
strtotime() 函 数 认识 的 英文 单词 还 有 很 多 ， 在 这 里 大 致 给 分 一 下 类 。 

口 月 份 名 称 : 从 一 月 到 十 二 月 的 英文 单词 以 及 它们 的 缩写 。 

口 星期 中 的 每 一 天 : 从 周一 到 周 日 的 英文 单词 以 及 它们 的 缩写 。 

口 时 间 单 位 : 年 (year) 、 月 (month) 、 日 (day) 、 星 期 (week) 、 小 时 Chour) 、 
分 (minute) 和 秒 (second) 等 。 

口 常用 表示 时 间 关 系 的 词语 : 之 后 (ago)、 现 在 (now) 、 上 个 (last) 、 下 个 (next) 、 
这 个 (this) 、 明 天 (tomorror) 和 昨天 (yesterday) 。 

口 加 减 号 。 


$importantDate = strtotime ("now + 15 hours"); // 输 出 15 小 时 后 的 时 间 截 


au wN 
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口 所 有 的 数字 。 
口 时 区 : 如 gmt (格林 威 治 时 间 ) 、pdt (太平洋 时 间 ) 和 bjt (北京 时 间 〉 等 。 

这 两 个 函数 都 比较 简单 ， 没 有 太 多 复杂 的 参数 ， 下 面 介绍 的 这 个 函数 参数 比较 多 。 大 
家 在 使 用 的 时 候 务 必要 仔细 。 

【 例 $.22】 存储 指定 时 间 到 变量 中 (二) : mktimeO。 

出 $importantDate = mktime(0, 0, 0, 1, 15, 2003); 

2 echo $importantDate; 

函数 mktime0 一 共有 六 个 参数 ， 分 别 为 小 时 、 分 、 秒 、 月 、 日 和 年 。 例 5.20 中 存储 在 
变量 $importantDate 中 的 日 期 如 果 用 strtotime0 函 数 的话 ， 就 可 以 写成 如 下 的 形式 : 


3 $importantDate = strtotime("January 15 2003"); 


5.5 判别 真 假 一 一 布尔 型 数据 


布尔 型 数据 分 为 两 类 。 

(1) true: 表示 事实 成 立 ， 为 真 ， 也 可 以 写成 true。 

(2) false: 表示 事实 不 成 立 ， 为 假 ， 也 可 以 写成 false。 

如 果 将 两 个 变量 进行 比较 , 并 根据 判断 的 结果 来 执行 某 些 操作 的 话 , 我 们 可 以 这 么 写 ， 
如 例 5.23 所 示 。 

【 例 5.23】 比较 变量 大 小 。 


<?php 

2 $numl = 2; 

3 Snum2 = 5; 

4 if ($numl == S$num2) { // 判 断 变量 Snuml 和 变量 Snum2 是 否 相 等 
区 echo '$numl 等 于 $num2'; 

6 } else { 

echo '$numl 不 等 于 $num2'; 

8 

9 ?> 


在 脚本 的 第 4 行 ,使 用 “==” 对 两 个 变量 的 值 进行 的 比较 。 由 于 变量 Snuml 和 变量 Snum2 
的 值 并 不 相等 ， 因 此 “$numl = Snum2” 的 值 为 false， 也 就 是 说 变量 ynuml 和 变量 Snum2 
相等 的 判断 不 成 立 。 这 时 ， 系 统 会 输入 第 二 句 echo 语句 ， 也 就 是 “$numl 不 等 于 Snum2”。 
当然 ， 其 他 类 型 的 变量 也 可 以 转换 成 布尔 型 变量 。 如 下 的 变量 转换 成 布尔 型 变量 时 ， 
其 值 为 false: 
整 型 值 0 ( 零 ) ， 
浮 点 型 值 0.0 ( 零 ) ， 
空白 字符 串 和 字符 串 "0"， 
没有 成 员 变 量 的 数组 ， 
没有 单元 的 对 象 〈 仅 适用 于 PHP 4) ， 
特殊 类 型 NULL (包括 尚未 设 定 的 变量 ) 。 
除 上 述 这 些 变量 之 外 ， 其 他 类 型 的 变量 在 转换 成 布尔 型 变量 之 后 ， 其 值 为 true。 


BBGQGDSea 
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5.6 实战 练习 : 计算 税 后 收入 


5.6.1 背景 介绍 
在 这 一 节 里 ， 我 们 编写 一 段 脚本 来 帮助 生活 在 北京 地 区 的 人 们 计算 其 税 后 收入 。 按 照 
北京 市 政府 的 规定 ， 城 镇 职工 需 按 月 缴纳 住房 公积金 和 社会 保险 ， 并 在 其 当月 的 工资 中 扣 
除 。 扣 除 后 剩余 的 工资 如 果 超 过 个 人 所 得 税 征 税 标准 的 ， 需 要 按照 个 人 所 得 税 征 税 标准 征 
收 个 人 所 得 税 。 表 5-4 所 示 列 出 了 在 计算 个 人 所 得 税 前 需 扣除 的 住房 公积金 和 社会 保险 的 
比例 。 表 5-5 所 示 列 出 了 国家 现行 的 个 人 所 得 税 征收 比率 。 
表 5-4 北京 地 区 城镇 职工 住房 公积金 和 社会 保险 占 月 收入 比例 


用 人 单位 缴纳 比例 
住房 公积金 
养老 保险 
[医疗 人 险 ”| gt3 | 
| 工作 全 险 0 
| 大业 保险 | 02% 
生育 保险 


表 5-5 国家 现行 个 人 所 得 税 税率 表 


社会 保险 


全 月 应 纳税 所 得 额 
x(%) | 这 从 
级 数 仿 税 级 下 未 会 税 级 证 税率 (%)| 速算 扣除 数 
1 不 超过 1500 元 的 不 超过 1455 元 的 3 0 
2 超过 1500 元 至 4500 元 的 部 分 超过 1455 元 至 4155 元 的 部 分 10 105 
3 超过 4500 元 至 9000 元 的 部 分 超过 4155 元 至 7755 元 的 部 分 20 555 
4 超过 9000 元 至 35000 元 的 部 分 超过 7755 元 至 27255 元 的 部 分 | 《25 1005 
5 超过 35000 元 至 55000 元 的 部 分 超过 27255 元 至 41255 元 的 部 分 | ”30 2755 
6 超过 55000 元 至 80000 元 的 部 分 超过 41255 元 至 57505 元 的 部 分 | 。 35 5505 
超过 80000 元 的 部 分 超过 57505 元 的 部 分 45 13505 
在 这 两 张 表 中 ， 只 需要 关注 底 色 为 灰色 的 部 分 就 可 以 了 。 现 在 我 们 开始 搭建 税 后 收入 
计算 器 。 同 前 两 章 的 实战 练习 一 样 ， 先 来 搭建 用 户 界面 : 
在 图 5-11 的 左 侧 有 一 个 表单 ， 供 用 户 输入 姓名 、 性 别 和 税 前 收入 。 当 用 户 填 好 表单 ， 


并 单 击 “ 计 算 税 后 收入 ”按钮 后 ， 在 表单 的 右 侧 会 出 现 一 张 报表 。 其 中 详细 地 列 出 了 该 用 
户 应 该 缴纳 的 五 险 一 金 和 个 人 所 得 税 的 金额 。 
我 们 一 起 来 看 看 这 是 如 何 实现 的 吧 。 


5.6.2 ”实现 过 程 
先 来 看 HIML 部 分 : 
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税 后 收入 计算 器 


姓名 张 晓 二 
性 别 : 一 男 器 女 
税 前 收入 : 25687| 


应 纳税 所 得 : 19,981.49 
应 缴 税 款 : 3,115.37 
税 后 收入 : 16,866.11 


@ Intermet | Protected Mode: Off 


图 5-11 税 后 收入 计算 器 用 户 界面 


<!DOCTYPE html> 
<html> 
<head> 
<meta charset="UTF-8"> 
<title> 税 后 收入 计算 器 </title> 
<style type="text/css"> 
body { 
font-size:small; 


1 


form div { 
line-height:35px; 
margin:5px Spx 5px Spx 
} 


.form-container { 
width:30%; 
float: left; 
background-color:#cccccc; 


} 


.info-container { 
width:55%; 
float:left; 


margin-left:10px; 


.info-container p, ul li { 
line-height:12px; 
margin—left:100px; 
font-family:"Lucida Sans Unicode"; 


上 


.info-container h3 { 
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text-align:center; 


| 
margin-left: 10px; 
} 


</style> 
</head> 
<body> 
<h2> 税 后 收入 计算 器 </h2> 
< 
<div class="form-container"> 
<form action="?check" method="post"> 
<div> 
<label for="user"> 姓 名 : </label> 
<input type="text" size="10" name="user"> 
</div> 
<div> 
<label for="gender"> 性 别 : </label> 
<input type="radio" name="gender" value="male" checked> 


<input type="radio" name="gender" value="female"> 


</div> 
<div> 
<label for="income"> 税 前 收入 : </label> 
<input type="text" size="15"” name="income"> 
元 
</div> 
<div> 
<button type="submit"> 计 算 税 后 收入 </button> 
</div> 
</form> 
</div> 
<div class="info-container" id="info"></div> 
// 用 于 存放 用 户 在 提交 表单 后 产生 的 各 项 数据 
</body> 
</html> 


在 这 段 HTML 代码 中 ， 使 用 CSS 和 DIV 定义 了 两 个 名 为 “form-container” 和 和 
“info-container” 的 DIV, 分 别 用 来 存放 表单 和 生成 的 纳税 报表 。 与 上 一 章 的 实战 演练 一 样 ， 
表单 的 action 属性 的 值 定义 为 “=?check”， 以 便 后 续 判断 用 户 是 否 提交 了 表单 。 

再 来 看 看 PHP 的 部 分 。 与 前 面 两 章 一 样 ， 我 们 需要 在 “</body>” 标 签 前 添加 PHP 
脚本 。 


<?php 
if(isset($ REQUEST['check'])){ // 判 断 用 户 是 否 提交 了 表单 
Susr = $ REQUEST['user']; // 获 取 用 户 名 
S$gen $_ REQUEST['gender']; // 获 取 用 户 性 别 


$inc = preg match('/^[0-9]*$/', $ REQUEST['income']) ? 
intval ($ REQUEST['income']) : ' 请 输入 您 的 税 前 收入 。'; 
/* 上 面 这 是 个 三 目 表达 式 ， 问 号 前 为 判断 条 件 ， 问 号 后 以 中 间 的 冒号 为 界 ， 
当 判 断 条 件 为 真 时 ， 用 冒号 前 的 表达 式 为 变量 赋值 ， 若 判断 条 件 为 假 ， 则 
用 冒号 后 的 表达 式 为 变量 赋值 。 2 
/* 在 上 面 这 个 三 目 表达 式 中 ， 我 们 使 用 了 preg_match () 函数 对 用 户 输入 的 税 前 收入 


和 
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进行 验证 ， 关 于 preg_match () 这 个 函数 的 介绍 ， 请 见 第 7 章 。 *#/ 


if (is int($inc)) { 
// 由 于 变量 $inc 的 值 可 能 为 整数 ， 也 可 能 为 字符 串 。 因 此 需要 判断 一 下 


Szhuf = Since 0 2 // 计 算 住房 公积金 
$yang = $inc * 0.08; // 计 算 养老 保险 
Syili = $inc * 0.02 + 3; ”// 计 算 医 疗 保险 
$gong = $inc * 0.00; // 计 算 工 伤 保险 
$shiy = $inc * 0.002; // 计 算 失业 保险 
$shen = $inc * 0.00; // 计 算 生育 保险 


S$beforeTax = $inc - $zhuf - $yang - $yili - $gong - $shiy - $shen; 
/7 计算 应 税 所 得 

if($beforeTax > 3500 && $beforeTax <= 5000) { 
Stax = ($beforeTax-3500) * 0.03; 

} elseif ($beforeTax > 5000 && $beforeTax <= 8000) { 
Stax = ($beforeTax-3500) * 0.1 - 105; 

} elseif ($beforeTax > 8000 && $beforeTax <= 12500) { 
Stax = ($beforeTax-3500) * 0.2 - 555; 

} elseif ($beforeTax > 12500 && S$beforeTax <= 38500) { 
Stax = ($beforeTax-3500) * 0.25 - 1005; 

} elseif ($beforeTax > 38500 && S$beforeTax <= 58500) { 
Stax = ($beforeTax-3500) * 0.3 - 2755; 

} elseif ($beforeTax > 58500 && S$beforeTax <= 83500) { 
Stax = ($beforeTax-3500) * 0.35 - 5505; 

} elseif ($beforeTax > 83500) { 
Stax = ($beforeTax-3500) * 0.4 - 13505; 

} else { 
Stax = 0; 

} // 计 算 应 缴 税 款 


SafterTax = $beforeTax - S$tax; // 计 算 税 后 收入 


$msg = '<h3> 税 后 收入 报表 </h3>'; 
Smsg .= '<p> 姓 名 :; '.$usr.'</p>'; 
if($gen == "male') { 
Smsg .= '<p> 性 别 : 男 </p>'; 
} else { 
Smsg .= '<p> 性 别 : 女 </p>'; 
} 
$Smsg -= '<p> 税 前 总 收入 : ' .number format ($inc,2).'</p>'; 
// 格 式 化 输出 
Smsg .= '<p> 其 中 需 缴 纳 : </p>"'; 
Smsg -= '<ul><1i> 住 房 公积金 : ' .number format ($zhuf,2).'</1i>'; 
$msg -= '<1i> 养 老 保险 : ' .number format ($yang,2).'</1i>'; 
Smsg -= "<1i> 医 疗 保险 : ' .number format ($yili,2).'</1i>'; 
Smsg -= "<1i> 工 伤 保险 : ' .number format ($gong,2).'</1i>'; 
Smsg -= "<1i> 失 业 保 险 : ' .number format ($shiy,2).'</1i>'; 
Smsg -= '<1i> 生 育 保险 : ' .number format (S$shen,2) ."</1i></ul>'7 
if(Stax > 0){ // 根 据 应 缴 税 款 是 否 为 0， 向 浏览 器 输出 纳税 提示 
Smsg -= '<p> 应 纳税 所 得 : ' -number format ($beforeTax, 2) 
ee 
Smsg -= "<p> 应 缴 税 款 : ' .number format ($tax,2).'</p>"'; 
Smsg -= "<p> 税 后 收入 : <strong>' .number format 
($afterTax, 2) .'</strong></p>'; 
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else 
-= '<p> 您 无 需 缴纳 个 人 所 得 税 .</p>"'; 
} i ‘| 
Smsg = $inc; 
"<script language="javascript">document .getElementById 
("info") .innerHTML="". 
$msg.'"</script>'; // 向 HTML 中 的 ID 为 info 的 div 输出 税 后 收入 报表 

2 : 

上 面 这 段 代 码 虽 然 比较 长 ， 总 共 也 不 过 6 个 步 又。 它们 分 别 是 : 

(1) 获取 用 户 提交 Ne 

(2) 计算 应 税 所 得 ; 

(3) 计算 应 缴 税 款 ; 

(4) 计算 税 后 收入 ; 

(5) 生成 税 后 收入 报表 ; 

(6) 将 税 后 收入 报表 输出 到 页 面 的 指定 位 置 。 

在 这 6 个 步骤 中 ， 第 1 步 〈《 如 何 获取 用 户 提交 的 信息 )》 和 第 6 步 〈 如 何 将 生成 的 信息 
通过 JavaScript 输出 到 浏览 器 中 ) 与 第 4 章 实战 演练 的 内 容 相 同 。 如 果 大 家 看 着 代码 也 想 
不 起 来 如 何 实现 这 两 个 步骤 的 话 ， 可 以 返回 第 4 章 的 实战 演练 部 分 再 看 一 下 。 

在 这 里 我 们 重点 关注 一 下 第 2 步 到 第 5 步 的 内 容 。 

在 第 2 步 中 ， 使 用 $_REQUEST 常量 数组 获取 用 户 通 过 表单 提交 的 三 个 元 素 的 值 ， 并 
将 获取 到 的 内 容 分 别 赋值 给 Susr、$gen 和 S$inc 三 个 变量 。 这 三 个 变量 分 别 代表 着 用 户 输入 
的 姓名 、 性 别 和 税 前 收入 。 其 中 ， 在 为 变量 Sinc 赋值 时 ， 我 们 使 用 了 三 目 表 达 式 。 

所 谓 的 三 目 表 达 式 本 来 是 微软 开发 的 C 语言 中 特有 的 一 种 表达 式 。 由 于 PHP 脚本 语 
言 继 承 了 C 语言 的 大 部 人 ee 因此 三 目 表 达 式 在 PHP 中 也 是 可 以 使 用 的 。 

在 一 个 三 目 表达 式 中 ， 一 共有 三 个 子 表达 式 ， “判断 表达 式 ”、“ 判 断 为 真 时 
的 赋值 表达 式 ” 及 “判断 为 假 时 的 赋值 表达 式 ” 三 个 子 表达 趟 之 间 用 i 
分 隔 开 ， 格 式 如 下 : 


$var = expression 1 ? expression 2 : expression 3 


使 用 这 个 表达 式 为 变量 $var 赋值 时 ， 若 “expression 1” 的 值 为 真 ， 则 将 “expression 2” 
的 值 赋 给 变量 $var， 若 “expression 2” 的 值 为 假 时 ， 则 将 “expression 3” 的 值 赋 给 变量 $var。 

脚本 中 为 变量 $inc 赋值 使 用 的 三 目 表达 式 如 下 : 

$inc = preg match('/^[0-9]*$/', $ REQUEST['income']) ? 

intval($ _ REQUEST['income']) : ' 请 输入 您 的 税 前 收入 。'; 

在 这 个 表达 式 中 ， 我 们 使 用 preg_match0 函 数 来 判断 用 户 在 表单 的 “ 税 前 收入 ”文本 
框 中 输入 的 是 不 是 一 个 大 于 零 的 正 整 数 。 若 是 则 将 该 值 的 数据 类 型 由 字符 串 型 转换 成 整 型 ; 
若 不 是 则 将 变量 $inc 的 值 定义 为 “请 输入 您 的 税 前 收入 ”。 

有 的 同学 看 到 这 里 可 能 会 问 了 : 为 什么 不 用 大 于 和 小 于 号 来 判断 用 户 输入 的 值 呢 ?要 
知道 系统 通过 表单 传递 的 值 ， 其 类 型 都 是 字符 串 ， 为 了 避免 在 计算 过 程 中 因数 据 类 型 不 一 
致 而 发 生 错误 ， 我 们 最 好 使 用 正则 表达 式 来 对 页 面 间 传 递 的 数据 进行 验证 。 关 于 正则 表达 
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式 ， 大 家 可 以 看 看 第 7 章 的 相关 内 容 ， 在 这 里 就 不 再 袭 述 了 。 

由 于 这 个 三 目 表 达 式 ， 变 量 $inc 可 能 为 一 个 整 型 变量 ， 也 可 能 为 一 个 字符 串 型 变量 。 
基于 此 ， 我 们 使 用 了 一 个 直 语 句 用 来 判断 变量 $inc 的 数据 类 型 。 若 变量 $inc 为 一 个 整 型 变 
量 ， 则 继续 执行 第 3 步 至 第 5 步 的 内 容 。 若 变量 $inc 为 一 个 字符 串 型 变量 ， 则 略 过 第 3 步 
至 第 5 步 的 内 容 ， 直 接 向 浏览 器 输出 错误 提示 ， 要 求 用 户 输入 正确 的 税 前 收入 。 

有 了 对 数据 类 型 的 判断 ， 我 们 就 可 以 保证 在 执行 第 3 步 到 第 5 步 时 ， 不 会 因数 据 类 型 
的 问题 出 现 错误 了 。 

在 第 3 步 里 ， 我 们 按照 表 5-4 中 的 规定 先 计 算 了 在 税 前 应 该 扣除 的 五 险 一 金 ， 并 将 这 
些 数 据 存储 在 名 为 Szhuf、$yang、$yii、$gong、S$shiy 和 $shen 六 个 变量 中 ， 分 别 代表 住房 
公积金 、 养 老 保 险 、 医 疗 保险 、 工 伤 保 险 、 失 业 保 险 和 生育 保险 。 然 后 ， 我 们 从 税 前 收入 
减 去 这 六 个 变量 的 值得 到 应 税 所 得 ， 并 将 应 税 所 得 赋值 给 变量 SbeforeTax。 

在 第 4 步 里 ， 我 们 根据 计算 得 出 的 应 税 所 得 和 表 5-5 的 内 容 计 算 应 缴 税 款 。 按 照 现行 
的 个 人 所 得 税 税率 表 ， 可 以 很 方便 的 计算 出 不 同 档 收 入 对 应 的 应 缴 税 款 。 公 式 如 下 : 

应 缴 税 款 = 〈 应 税 所 得 -所 得 税 起 征 点 ) X 相应 档 位 的 税率 - 同 档 速算 扣除 数 

举 个 例子 ， 若 某 先 生 的 应 税 所 得 为 10,000 元 ， 现 行 个 人 所 得 税 的 起 征 点 为 3500 元 。 
则 该 先生 应 该 按照 第 三 档 标准 纳税 。 该 档 位 的 税率 为 0.20， 速 算 扣 除数 为 555 元 。 因 此 ， 
该 先生 的 应 缴 税 款 为 754 元 。 

按照 这 个 公式 ， 我 们 使 用 让 ..else 语句 在 脚本 中 列 出 了 七 个 档 位 的 计算 公式 及 使 用 这 
些 公式 的 条 件 。 系 统 会 根据 第 3 步 中 计算 出 的 应 税 所 得 自动 套用 这 七 个 档 位 的 计算 公式 中 
的 一 个 来 计算 应 缴 税 款 。 

在 计算 去 应 缴 税 款 后 ， 我 们 用 第 3 步 中 计算 出 的 应 税 所 得 减 去 应 缴 税 后 ， 即 可 得 到 税 
后 收入 。 

在 第 5 步 里 ， 我 们 开始 输出 计算 结果 到 报表 中 。 由 于 此 处 比较 复杂 ， 请 允许 我 在 下 方 
再 次 列 出 输出 报表 的 脚本 。 

Smsg = '<h3> 税 后 收入 报表 </h3>'; 

Smsg .= '<p> 姓 名 : '.$usr.'</p>'; 

if($gen == 'male') { 

Smsg .= '<p> 性 别 : 男 </p>"'; 


} else { 
Smsg .= '<p> 性 别 : 女 </p>"'; 
} 
Smsg .= '<p> 税 前 总 收入 : ' .number format ($inc,2).'</p>'; Wn 
Smsg .= '<p> 其 中 需 缴 纳 : </p>"'; 
Smsg .= '<ul><1i> 住 房 公积金 : ' .number format ($zhuf,2).'</1i>'; 
Smsg .= '<1i> 养 老 保险 : ' .number format ($yang,2).'</1i>'; 
Smsg .= '<1i> 医 疗 保险 : ' .number format ($yili,2).'</1li>'; 
Smsg .= "<1i> 工 伤 保险 : ' .number format ($gong,2).'</1i>'; 
Smsg .= "<1i> 失 业 保 险 : ' .number format ($shiy,2).'</1i>'; 
Smsg .= '<1i> 生 育 保险 : " -number format ($shen,2).'</1i></ul>'; 


if($tax > 0){ // 根 据 应 缴 税 款 是 否 为 0， 向 浏览 器 输出 纳税 提示 
Smsg "<p> 应 纳税 所 得 : ' -number format (SbeforeTax,2) - "</P> "7 
$msg "<p> 应 缴 税 款 : ' .number format (Stax,2) .'</p>"'; 
Smsg -= '<p> 税 后 收入 : <strong>" -number format (SafterTax,2) 
</strong></p> 
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else 

-= ' <p> 您 无 需 缴纳 个 人 所 得 税 .</p>"'; 

在 第 5 步 的 脚本 里 ， 先 定义 了 一 个 变量 Smsg， 并 将 其 值 定义 为 “<h3> 税 后 收入 报表 
</h3>”。 值 得 注意 的 是 ， 在 这 段 脚本 里 ， 除 了 在 定义 变量 Smsg 时 使 用 的 是 赋值 符 (=) 之 
外 ， 其 他 使 用 变量 $msg 的 地 方 都 在 赋值 符 前 加 了 一 个 点 〈.) ， 表 示 将 “=” 后 的 内 容 附加 
到 变量 $msg 中 。 这 个 点 其 实 就 是 我 们 在 5.3 节 里 学 到 的 连接 字符 串 的 运算 符 。 

在 定义 了 变量 gmsg 后 ， 我 们 将 获取 用 户 输入 的 “姓名 ”和 “性 别 ” 附 加 到 变量 中 ， 接 
着 把 我 们 计算 得 出 的 “ 税 前 收入 ”和 “五 险 一 金 ” 附 加 到 变量 中 。 最 后 根据 应 缴 税 款 是 否 
为 零 ， 来 决定 是 输出 “应 税 所 得 ”、“ 应 缴 税 款 ”和 “ 税 后 收入 ”， 还 是 输出 “您 无 需 缴 
纳 个 人 所 得 税 ” 这 样 一 名 提示 。 

在 向 变量 $msg 中 附加 这 些 信 息 时 , 使 用 了 一 些 HTML 代码 对 这 些 信 息 进行 了 格式 化 ， 
使 得 这 些 信 息 可 以 如 图 5-11 所 示 的 那样 有 序 输出 。 

至 此 ， 我 们 就 完成 了 税 后 收入 计算 器 脚本 的 编写 。 


5.7 习 题 


(1) 请 判断 如 下 脚本 的 运行 结果 : 
<?php 
$a = (int) 5.6; 
$b = (float) 5.6; 
$c = $a + Sb; 
echo $c; 
?> 
(2) 请 使 用 sprint0 函 数 输出 本 章 的 目录 ， 注 意 一 级 标题 与 二 级 标题 之 间 、 以 及 二 级 标 
题 与 三 级 标题 之 间 有 缩 进 。 
(3) 请 使 用 strtotimeO 函 数 和 date0 函 数 输 出 当前 日 期 前 25 天 的 日 期 、 前 一 个 月 的 日 
期 、 前 五 周 的 日 期 、 前 三 个 月 的 日 期 和 前 半年 的 日 期 。 
(4) 请 使 用 三 目 表 达 式 编写 一 段 脚本 。 
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数组 ? 什么 数组 ? 看 到 这 个 词 ， 很 多 同学 的 第 一 反应 就 是 这 个 。 难 不 成 是 一 组 数 吗 ? 
其 实 ， 还 真 可 以 把 数组 看 成 是 用 于 存放 一 组 数据 的 复杂 变量 。 比 如 ， 可 以 把 关于 用 户 的 一 
组 信息 存放 在 这 个 变量 $userInfo 下 。 存 放 完成 后 ， 可 以 很 方便 地 读 取 、 修 改 和 处 理 数组 中 
存放 的 数据 。 

在 本 章 里 ， 我 们 将 学 习 如 何 创建 、 修 改 、 复 制 和 使 用 数组 。 


6.1 多胎 胞 一 一 数组 的 声明 与 使 用 


在 PHP 的 概念 里 , 数组 有 着 十 分 重要 的 作用 。 本 节 将 学 习 如 何 创建 、 修改 和 删除 数组 。 
6.1.1 创建 数组 


如 同 通过 给 变量 赋值 来 创建 变量 一 样 ， 可 以 通过 给 变量 赋值 一 组 数组 来 创建 数组 。 例 
如 ， 可 以 使 用 下 面 的 语句 来 定义 一 个 名 为 $userInfo 的 数组 : 


$userInfo[1] = "Jim Green"; 


通过 这 句 脚 本， 一 个 名 为 SuserInfo 的 数组 就 创建 好 了 。 在 这 个 数组 中 ， 有 一 个 元 素 ， 
其 值 为 Jim Green。 接 下 来 ， 可 以 使 用 下 面 的 语句 为 这 个 数组 添加 新 的 元 素 : 

$userInfo[2] 

$userInfo[3] 

现在 ，S$userInfo 就 是 一 个 拥有 三 个 元 素 的 数组 了 。 

按照 这 样 的 说 法 ， 一 个 数组 可 以 被 看 做 是 一 个 键 值 对 (key-value pair) 的 列表 ， 信 息 
将 按照 以 下 的 结构 被 存储 起 来 : 


SarrayName ['key1l1'] = valuel; 
SarrayName ['key2'] = value2; 
SarrayName ['key3'] = value3; 


Wi 
"Male"; 


一 个 数组 可 以 包含 无 限 个 元 素 。 这 里 的 键 (key) 又 可 以 被 称 为 索引 值 。 在 很 多 的 编程 
语言 中 ， 数 组 的 索引 只 能 是 数字 形式 的 ， 而 在 PHP 等 少数 几 种 脚本 语言 和 编程 语言 中 ， 数 
组 的 索引 既 可 以 是 数字 形式 的 ， 也 可 以 是 字符 串 形式 的 。 为 了 说 明 这 个 问题 ， 我 们 使 用 下 
面 的 语句 来 重新 定义 一 下 $userInfo 这 个 数组 : 


SuserInfo["Name"] = "Jim Green™"; 
$userInfo["Age"] = "18"; 
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$userInfo["Gender"] = "Male"™; 


这 样 一 来 ， 我 们 就 建立 了 索引 和 值 之 间 一 一 对 应 的 关系 ， 并 且 也 便于 其 他 程序 员 阅 读 
这 段 脚本 。 当 然 ， 如 果 像 一 个 用 来 储存 价格 的 数组 ， 可 以 使 用 更 为 快捷 的 方式 定义 它 : 


$price[] = 19.25; 
$price[] = 22.35; 
$price[] = 133.24; 


在 定义 数组 $price 这 个 数组 时 ， 我 们 没有 指定 索引 值 。 当 PHP 处 理 引擎 在 处 理 到 这 样 
的 数组 时 会 自动 为 其 分 配 数字 形式 的 索引 值 。 不 过 自动 分 配 的 索引 值 是 从 0 开始 的 。 也 就 
是 说 ， 上 面 这 段 脚本 等 价 于 : 


$price[0] = 19.25; 
$price[1] = 22.35; 
$price[2] = 133.24; 


除了 通过 直接 赋值 的 方式 创建 数组 之 外 ， 我 们 还 可 以 通过 array0 函 数 来 创建 数组 。 例 
如 ， 要 创建 一 个 名 为 Sfruits 的 数组 ， 那 么 可 以 用 下 面 的 语句 来 声明 这 个 数组 : 


$fruits = array ("Apple", "Orange", "Banana", "Grapefruit"); 


那么 $fruits[3] 的 值 就 是 Grapefruit 了 。 这 也 说 明了 , 通过 array0 函 数 创建 的 数组 中 各 元 
素 的 索引 值 也 是 由 系统 从 0 开始 自动 分 配 的 。 那 么 ， 我 们 可 不 可 以 在 使 用 aray0 函 数 的 同 
时 ， 使 用 字符 串 形式 的 索引 值 呢 ? 为 了 实现 这 个 愿望 ，PHP 提供 了 如 下 的 方式 : 

$arrayName = array('keyl' => ‘valuel', 

'key2' => "value2'， 

'key3' => 'value3', 

'key4' => 'value4'); 

在 使 用 aray0 函 数 时 ， 我 们 使 用 “=>” 符 号 来 定义 键 值 对 。 这 个 符号 的 左边 是 数组 元 
素 索 引 ， 而 右边 是 元 素 值 。 每 两 个 键 值 对 之 间 用 逗号 阳 开 。 为 了 方便 脚本 维护 ， 也 可 以 在 
用 于 隔 开 键 值 对 的 逗号 后 回 车 。 这 样 做 不 会 影响 PHP 处 理 引 擎 对 脚本 的 处 理 。 

除了 可 以 在 array0 函 数 中 定义 字符 串 形式 的 索引 之 外 ， 还 可 以 通过 为 第 一 个 元 素 指定 
不 为 0 的 数字 索引 来 为 数组 中 的 元 素 自动 分 配 数字 形式 的 索引 ， 例 如 : 


StopNations = array('12' => 'India', 'Pakistan','Israel','Budan'); 


通过 这 条 语句 定义 的 数组 $topNations[2] 的 值 为 未 定义 ， 而 StopNations[14] 为 Israel。 
另外 , 如 果 需 要 创建 一 个 以 连续 值 为 元 素 的 数组 , 则 可 以 使 用 rangeO 函 数 来 定义 数组 。 
比如 : 


$year = range(2002, 2012); 


这 条 语句 相当 于 : 
Syear[0] = 2002; 
$year[1] = 2003; 


$year[10] = 2012 


类 似 的 ， 我 们 还 可 以 使 用 range0 来 定义 数组 Salphabet: 


$alphabet = range('a', '2z'); 
这 条 语句 相当 于 : 
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S$alphabet [0] a 


S$alphabet[1] = 'b'; 
S$alphabet[2] = 'c'; 
$alphabet[24] = 'y'; 
$alphabet[25] = 'z'; 


6.1.2 ”查看 数组 


在 创建 了 数组 之 后 ， 可 以 使 用 var_ dump0 函 数 和 print_rO 函 数 来 查看 数组 的 结构 。 前 
者 和 后 者 的 区 别 在 于 ， 前 者 提供 了 更 为 丰富 的 关于 数组 元 素 的 信息 ， 而 后 者 与 之 相 比 略 显 
简单 。 
现在 ， 我 们 分 别 使 用 这 两 个 函数 来 查看 在 上 一 节 一 开始 就 创建 了 的 数组 SuserInfo 的 结 
也 就 是 如 例 6.1 所 示 的 脚本 。 

【 例 6.1】 查看 数组 结构 (一 )。 


入 <?php 


构 


Ea $userInfo[0] = "Jim Green"; 

3 $userInfo[1] = 18; 

4 SuserInfo[2] = "Male"; 

5 

6 echo "The following displays the output of the var dump 
function:<br>"; 

echo "<pre>"; 

8 var dump ($userInfo); 

9 echo "</pre>"; 

10 

时 echo "<br>The following displays the output of the print r 
function:<br>™> 

2 echo "<pre>"; 

3 Print r($userInfo); 

14 echo "</pre>"; 

1 


得 到 的 结果 如 图 6-1 所 示 。 


Bmttps/ ww graatwalltaa/chaptefiEVB6-OLphp | [5 | < ny | 
将 Favorites | 菩 htpVyiwwwgrestwalltca/chaptctG/E6-0Lphp 堆 二 国 世 Di 各 WisSew=Tk 和 


The following dicplays the output of the var_dump functicn: 
array(3) [ 
加 = 


atring (9) “Jin Crasn 


[=> 
tring tt) “Kale” 
1 


The following displays the output of the print_r function: 
Array 
‘ 

[ol = 


rs 
[2 = Tale 


Done @ enet Prtected Mode OF 三 -10% ~ 


图 6-1 查看 数组 结构 (一) 
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按照 图 6-1 所 示 , var_dump() 函 数 除了 给 出 print_r0 函 数 给 出 的 数组 各 元 素 键 值 对 之 外 ， 
还 提供 了 如 下 信息 : 
口 数组 元 素 个 数 ; 
口 数组 元 素 的 数据 类 型 ; 
口 若 某 元 素 是 字符 串 型 的 数据 ， 还 输出 了 该 字符 串 的 长 度 。 

如 果 在 数组 中 使 用 字符 串 形 式 的 索引 ， 那 么 在 查看 数据 结构 时 ， 键 值 对 中 出 现 的 索引 
就 会 被 包含 在 一 对 引号 和 一 对 方 括号 之 间 ， 比 如 例 6.2 所 示 。 

【 例 6.2】 查看 数组 结构 (二) 。 


重 <?php 

2 SuserInfo["Name"] = "Jim Green"; 

3 $userInfo["Age"] = 18; 

4 SuserInfo ["Gender"] = "Male"; 

5 

6 echo "The following displays the output of the var dump 


function:<br>"; 


当 echo "<pre>"; 

8 var dump ($userInfo); 

a echo "</pre>"; 

10 

和 echo "<br>The following displays the output of the print r 
function:<br>"; 

生父 echo "<pre>"; 

3 Print r($userInfo); 

14 echo "</pre>"; 


US 


在 使 用 var_dump0 函 数 和 print rO 函 数 查看 数组 SuserInfo 的 结构 时 ， 显 示 的 结构 如 图 
6-2 所 示 。 


ee 


SR - EE 
帘 Favorites 臣 http://www.greatwalltea/chapter06/Ex6-02.php 全 -> 7 四 及 v pagev Ssfetyv Toosv 固 ” 


The following displays the output of the var_dump function: 


array(3) { 
["Nane“]=> 
string (9) “Jim Green” 
["Age”]=> 
int (18) 
[Gender”]=> 
string (4) "Male” 


The following displays the output of the print_r function: 


Array 
《 


[Nane] => Jin Green 
[Age] => 18 
[Gender] => Male 


全 Intemet| Protected Mode Off 红 ~ 胞 100% ~ 


图 6-2 查看 数据 结构 (二) 
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6.1.3 修改 数组 


数组 一 旦 创建 , 其 元 素 的 值 是 可 以 随时 修改 的 , 就 像 可 以 随时 修改 某 个 变量 的 值 一 样 。 
同样 ， 也 可 以 随时 改变 数组 中 元 素 个 数 ， 并 复制 整个 数组 。 例 如 ， 有 一 个 数组 $capitals， 
可 以 使 用 以 下 的 语句 来 定义 这 个 数组 : 

$capitals = array ("SD" => "Jinan", 

"HEB" => "Baoding", 

"HEN" => "Zhengzhou", 

"HB" => "Wuhan"， 

"HN" => "Changsha", 

"GD" => "Guangzhou", 

"GX" => "Nanning"); 

在 这 个 数组 中 ， 定 义 了 中 国 若干 省 份 的 省 会 城市 ， 其 中 河北 省 HEB) 的 省 会 被 误 定 
义 成 了 保定 市 。 如 果 需 要 将 其 修改 为 正确 的 石家庄 市 ， 则 可 以 使 用 下 面 这 条 语句 : 

S$capitals["HEB"] = "Shijiazhuang"; 

如 果 需 要 向 数组 $capitals 中 添加 山西 省 SX) 省 会 太原 和 陕西 省 SAX) 省 会 西安 ， 
则 可 以 使 用 下 面 这 两 条 语句 : 


Scapitals["SX"] = "Taiyuan"; 
$capitals["SAX"] = "Xi'an"; 


如 果 需 要 为 数组 $capitals 做 一 个 备份 ， 即 数组 $capitals 现 有 的 元 素 备 份 到 数组 
$capitalsBackup 中 ， 则 可 以 使 用 下 面 这 条 语句 : 
$capitalsBackup = $capitals; 


到 此 ， 我 们 一 共 创 建 了 两 个 数组 ， 一 个 为 $capitals， 另 一 个 为 $capitalsBackup。 现 在 使 
用 $var_dump 来 查看 一 下 这 两 个 数组 的 结构 ， 如 图 6-3 所 示 。 


| 和 httpy/wwwgreatwaltearchapte06/E5-03ohp 


| 帘 Favortes | @ pitp/ wwew grestwall tea/ chapterO/E6-03.php 寅 Fontee | 大 htp//wwworeatwalltea/chapted /E603php | 


a The structure of $capitalsBackup is as follows: 


array(9) { 


array(9) { 
[SD"]=> 人 区 


string (5) “Jinan” 
[HEE”]=> 


string (5) “Jinan” 
]- CHEB-]=》 

strire (12) Shijiazhuang” 
T"HERN]= 


string (12) “Shijiazhuang” 
> [HEN"]=> 

“Zhengzhour Tool 
5) “Waban’” string (5) “Wuhan” 
“Changsha” string (8) “Changshar 
“Guangzhou” 


“Namming” 


“Taiyuamr string (1) “Taiyuan” 
> 


strine (5) “Ki an” 


Done @r 


图 6-3 修改 数组 以 及 复制 数组 元 素 
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如 果 需 要 移 除数 组 $capitalsBackup 中 的 广东 省 GD) 的 省 会 ， 可 以 使 用 下 面 的 语句 : 


unset ($capitalsBackup["GD"]); 


当 PHP 引擎 执行 了 这 条 语句 之 后 ,ScapitalsBackup 这 个 数组 中 元 素 的 个 数 就 由 原来 的 
9 个 变 成 如 图 6-4 所 示 的 8 个 元 素 了 。 
我 们 可 以 再 次 使 用 var dump0 函 数 检查 一 下 ， 如 图 6-4 所 示 。 


和 | 


e 9 FIDS 辐 本 四 


寓 Favorites 。 赔 http://www.greatwalltea/chapter06/Ex6-04.php 价 ~ ”可 哆 ~ pagev safeyv Toosv 避 - 


了 


The structure of $capitalsBackup is as follows: 


array(8) 上 
SD ]=> 


string (5) “Jinan” 
["HEB"]=> 

string (12) “Shijiazhuang" 
["HEN"]=> 

string (9) "Zhengzhou” 


图 6-4 移 除 数组 元 素 后 


既然 可 以 用 unset0 函 数 来 删除 变量 和 数组 元 素 ， 那 是 不 是 也 可 以 用 它 来 删除 整个 数组 
呢 ? 答案 是 肯定 的 。 比 如 ， 现 在 我 们 觉得 ScapitalsBackup 这 个 数组 没有 什么 作用 了 ， 想 删 
除 它 ， 那 么 可 以 执行 下 面 的 语句 : 


unset ($capitalsBackup); 


当 再 次 使 用 var_dump0 函 数 显示 数组 结构 时 ， 只 会 得 到 如 下 所 示 的 错误 提示 : 

Notice: Undefined variable: capitalsBackup in C:\greatwall\chapter06\ 

Ex6-05.php on line 31 

提示 信息 指出 capitalsBackup 是 个 未 定义 变量 。 由 此 可 知 ，$capitalsBackup 已 经 不 存在 
了 。 有 的 同学 可 能 还 会 有 疑问 ，$capitalsBackup 不 是 个 数组 吗 ， 为 什么 这 里 说 它 是 个 未 定 
义 变量 呢 ? 

对 于 这 个 问题 ， 我 们 可 以 做 如 下 解读 : 

S$capitalsBackup 在 未 删除 之 前 的 确 是 个 数组 ， 但 是 删除 之 后 ， 它 都 不 存在 了 ，PHP 处 
理 引 擎 怎么 会 知道 它 是 数组 呢 ， 自 然 也 就 把 它 当 成 变量 了 。 

如 果 你 对 这 个 解答 不 太 满意 ， 那 就 看 看 本 章 的 开头 吧 。 在 那里 我 曾 说 过 这 么 一 句 话 ， 
那 就 是 : “其 实 ， 我 们 还 真 可 以 把 数组 看 成 是 用 于 存放 一 组 数据 的 复杂 变量 。” 

数组 其 实 就 是 一 种 特殊 的 变量 。 提 示 信 息 中 称 其 为 变量 ， 理 解 起 来 还 真 没有 什么 障 
碍 啊 。 

最 后 ， 我 们 把 这 一 节 里 使 用 到 的 脚本 整合 一 下 ， 放 在 下 面 ， 供 大 家 参考 。 

【 例 6.3】 向 数组 中 添加 元 素 、 修 改 数组 元 素 和 复制 数组 。 


和 <?php 

$capitals = array ("SD" => "Jinan"， 
3 "HEB" => "Baoding", 

4 "HEN" => "Zhengzhou"， 
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3 "HB" => "Wuhan™, 

6 "HN" => "Changsha", 
y "GD" => "Guangzhou", 
8 "GX™" => "Nanning"); 


9 

10 // 把 河北 的 省 会 改 成 石家庄 

1 $capitals["HEB"] = "Shijiazhuang"; 

下 2 

13 // 添 加 太原 和 西安 两 个 省 会 

14 $capitals["SX"] = "Taiyuan"7 

二 $capitals["SAX"] = "Xi'an™"; 

16 

小 贡 $capitalsBackup = $capitals; // 备 份 数组 

18 

19 echo "The structure of \$capitals is as follows:<br>"; 
20 echo "<pre>"; 

pl var dump ($capitals); 

2 echo "</pre>"; 

23 

24 echo "<br>The structure of \$capitalsBackup is as follows:<br>"; 
2 echo "<pre>"; 

26 var dump ($capitalsBackup); 

27 echo "</pre>"; 

28 ?> 

【 例 6.4】 删除 数组 中 的 指定 元 素 。 

二 <?php 

之 $capitals = array ("sD" => "Jinan", 

3 > "Baoding", 

4 => "Zhengzhou", 

5 "Wuhan™", 

6 "Changsha", 

2 "Guangzhou", 

8 "Nanning"); 

9 

10 $capitals["HEB"] = "Shijiazhuang"; // 把 河北 的 省 会 改 成 石家庄 
uel // 添 加 太原 和 西安 两 个 省 会 

2 $capitals["SX"] = "Taiyuan"7 

于 3 $capitals["SAX"] = "Xi'an"; 

14 

25 $capitalsBackup = $capitals; // 备 份 数组 

16 

于 这 unset ($capitalsBackup["GD"]); // 移 除 广 东 省 的 省 会 
18 

9 echo "The structure of \$capitals is as follows:<br>"; 
20 echo "<pre>"; 

忆 出 Var dump ($capitals); 

区 echo "</pre>"; 

23 

24 echo "<br>The structure of \$capitalsBackup is as follows:<br>"; 
25 echo "<pre>"; 

26 Var dump ($capitalsBackup); 

Eb echo "</pre>"; 

过 8 

【 例 6.5】 删除 整个 数组 。 

‘ <?php 

多 $capitals = array ("sD" => "Jinan"， 


。98 。 


第 6 章 抱团 效应 


数组 


3 ”HEB”=> "Baoding", 

4 "HEN" => "Zhengzhou", 
5 "HB" => "Wuhan™, 

6 "HN" => "Changsha"， 

下 "GD"” => "Guangzhou", 
8 "GX" => "Nanning"); 


9 

0 $capitals["HEB"] = "Shijiazhuang"; 

11 

到 $capitals["SX"] = "Taiyuan"7 

3 $capitals["SAX"] = "Xi'an™"; 

14 

15 $capitalsBackup = $capitals; // 备 份 数 组 

16 

EB unset ($capitalsBackup); // 删 除数 组 

18 

9 echo "The structure of \$capitals is as follows:<br>"; 
20 echo "<pre>"; 

2 Var dump ($capitals); 

22 echo "</pre>"; 

2 

24 echo "<br>The structure of \$capitalsBackup is as follows:<br>"; 
A echo "<pre>"; 

26 var dump ($capitalsBackup); 

2 echo "</pre>"; 

2 > 


6.2 ” 排 排 坐 数组 的 遍历 、 排 序 与 比较 


通过 上 一 节 的 学 习 ， 我 们 知道 了 如 何 创建 数组 、 查 看 数组 元 素 、 修 改 数 组 元 素 、 向 数 
组 中 增加 数组 、 删 除数 组 元 素 以 及 删除 数组 ， 大 体 上 了 解 了 什么 是 数组 。 在 本 节 里 ， 我 们 
将 继续 上 一 节 的 学 习 ， 了 解 如 何 遍 历数 组 元 素 、 给 数组 元 素 排 序 和 比较 任意 两 个 数组 。 


6.2.1 如 何 遍历 数组 中 的 元 素 


“遍历 ”这 个 词 看 上 去 可 能 比较 高 级 。 所 谓 “遍历 ”， 也 就 是 依次 对 数组 中 的 每 个 元 
素 分 别 做 些 操作 。 例 如 ， 如 果 需 要 打印 数组 中 的 每 一 个 元 素 或 者 把 数组 中 的 每 一 个 元 素 分 
别 存储 进 数 据 库 中 ， 这 都 需要 用 到 数组 的 遍历 。 在 本 小 节 里 ， 我 们 将 学 习 两 个 遍历 数组 的 
方式 : 手动 遍历 和 自动 遍历 。 

1. 手动 遍历 

所 谓 手动 遍历 ， 就 是 指使 用 脚本 控制 系统 指针 从 指定 数组 的 第 一 个 元 素 开 始 ， 一 个 一 
个 的 向 前 移动 。 每 移动 一 步 ， 就 做 一 些 操作 。 这 里 ， 可 以 把 数组 看 成 是 一 张 表 ， 而 每 个 数 
组 元 素 就 是 一 个 表 项 ， 在 表 的 左边 有 一 个 可 以 上 下 浮动 的 游标 。 我 们 可 以 用 手 滑动 游标 来 
指定 不 同 的 数组 元 素 。 

PHP 提供 了 很 多 简化 手动 遍历 数组 元 素 的 函数 ， 可 以 先 来 认识 一 下 它们 。 

口 current($arrayName): 指 的 是 系统 指针 当前 指向 的 数组 元 素 ， 又 称 为 当前 指针 。 

口 next($arrayName): 指 的 是 当前 指针 指向 元 素 的 下 一 个 数组 元 素 。 
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口 prev($arrayName): 指 的 是 当前 指针 指向 元 素 的 上 一 个 数组 元 素 。 

口 end($arrayName): 指 的 是 当前 数组 的 最 后 一 个 数组 元 素 。 

在 对 一 个 数组 做 手动 遍历 操作 时 ， 除 非 你 已 经 移动 过 系统 指针 ， 和 否则 当前 指针 为 该 数 
组 的 第 一 个 元 素 。 如 果 之 前 已 经 移动 过 系统 指针 ， 而 又 想 让 当前 指针 为 该 数组 的 第 一 个 元 
素 ， 则 可 以 在 遍历 数组 之 前 使 用 reset0 函 数 ， 如 : 


reset ($arrayName); 


如 果 使 用 手动 遍历 的 方法 来 遍历 数组 ， 我 们 需要 用 到 一 个 赋值 语 铅 和 一 个 输出 语句 来 
输出 数组 中 的 指定 元 素 的 值 。 现 在 ， 还 是 使 用 上 一 节 中 定义 的 数组 Scapitals 来 举 个 例子 。 

【 例 6.6】 手动 遍历 数组 元 素 。 

1 $capitals = array("SD" => "Jinan", 

"HEB" => "Baoding", 

3 "HEN" => "Zhengzhou", 

4 "HB" => "Wuhan", 
5 "HN" => "Changsha", 
6 
六 
8 


"GD" => "Guangzhou", 
"GX" => "Nanning"); 


9 “// 和 输出 当前 指针 指向 的 数组 元 素 
10 S$value = current($capitals); 
11 echo "The current pointer is <b>$value</b>.<br>"; 


13 “// 输 出 当前 指针 指向 的 数组 元 素 的 下 一 个 元 素 
14 S$value = next($capitals); 
15 echo "The next pointer is <b>$value</b>.<br>"; 


17 // 输 出 数组 的 最 后 一 个 元 素 
18 S$value = end($capitals); 
19 echo "The last pointer is <b>$value</b>.<br>"; 


21 // 输 出 当前 指针 指向 数组 元 素 的 上 两 个 元 素 

22 S$value = prev($capitals); 

23 S$value = prev($capitals); 

24 echo "The pointer two elements ahead of the current pointer is 
<b>$value</b>.<br>"; 


这 段 脚本 运行 的 结果 如 图 6-5 所 示 。 


| 
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The current pointer is Jinan. 
The next pointer is Baoding. 
| The last pointer is Nanning. 
The pointer two elements ahead of the current pointer is Changsha. 


图 6-5 手动 遍历 数组 元 素 


通过 结果 可 知 ， 第 一 组 赋值 和 输出 语句 的 结果 显示 当前 系统 指针 指向 数组 Scapitals 的 
第 一 个 元 素 〈 即 山东 省 的 省 会 济南 ) 。 第 二 组 赋值 和 输出 语句 的 结果 显示 当前 系统 指针 指 
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向 当前 指针 的 下 一 个 元 素 〈 即 河北 省 省 会 保定 ) 。 第 三 组 赋值 和 输出 语句 的 结果 显示 当前 
系统 指针 指向 数组 $capitals 的 最 后 一 个 元 素 〈 即 广西 区 首府 南宁 ) 。 在 第 四 组 赋值 和 输出 
语 名 中， 使 用 了 两 次 prev0 函 数 ， 也 就 是 连续 两 次 向 前 移动 了 系统 指针 ， 通 过 图 6-5 所 示 
中 显示 的 结果 可 以 看 出 ， 当 前 系统 指针 指向 了 数组 Scapitals 的 倒数 第 三 个 元 素 〈 即 湖南 省 
省 会 长 沙 ) 。 

由 此 可 见 , 我 们 可 以 通过 连续 使 用 nextO0、prevO 和 endO 函 数 来 移动 系统 当前 指针 到 数 
组 中 的 任 一 位 置 ， 然 后 使 用 赋值 和 echo 语句 来 输出 当前 指针 。 

虽然 手动 遍历 数组 元 素 看 上 去 十 分 简单 ， 但 是 对 于 动 辆 数 万 元 素 的 数组 来 说 却 只 能 望 
洋 兴叹 了 吧 。 没 有 关系 ， 我 们 可 以 使 用 自动 遍历 数组 元 素 的 方法 来 对 付 庞大 的 数组 。 


2. 自动 遍历 


其 实 所 谓 的 自动 遍历 就 是 使 用 在 过 历数 组 时 使 用 循环 语句 foreach。 使 用 该 语句 可 以 让 
系统 一 个 接着 一 个 的 输出 数组 中 的 每 个 元 素 ， 一 次 输出 一 个 。 语 句 结构 如 下 : 


foreach ($arrayName as $key => $value) 


statements 


li 


其 中 ， 

口 arrayName 指 的 是 需要 遍历 的 数组 名 称 ， 在 例 6.7 中 ， 我 们 将 继续 使 用 6.1.3 小 节 
中 定义 的 数组 $capitals。 

口 key 指 的 是 用 于 存储 数组 索引 的 变量 名 。 这 个 参数 是 可 选 的 。 在 例 6.7 中 ， 我 们 将 
使 用 变量 $province 来 存储 数组 索引 。 

口 value 指 的 是 用 于 存储 数组 元 素 值 的 变量 名 。 在 例 6.7 中 ,我 们 将 使 用 $capital 来 存 
储 数组 元 素 的 值 。 

下 面 ， 我 们 来 看 一 段 脚 本 ， 学 习 一 下 如 何 自动 遍历 指定 数组 。 

【 例 6.7】 自动 遍历 数组 元 素 。 


1 $capitals = array("SD"” => "Jinan", 

2 "HEB" => "Shijiazhuang", 
3 "HEN" => "Zhengzhou", 

4 "HB" => "Wuhan", 

5 "HN" => "Changsha", 

6 "GD" => "Guangzhou", 

了 "GX" => "Nanning"); 

8 


9 ”// 使 用 foreach 语句 来 遍历 数组 
10 foreach ($capitals as $province => $capital) { 
echo "The capital of S$province is $capital.<br>"; 


4 
这 段 脚本 的 运行 结果 如 图 6-6 所 示 。 
通过 如 图 6-6 所 示 的 结果 ， 可 以 知道 ， 变 量 Sprovince 存储 了 系统 当前 指针 指向 元 素 的 
索引 ， 而 Scapital 则 存储 了 系统 当前 指针 指向 元 素 的 值 。 通 过 foreach 语句 一 条 一 条 地 将 数 
组 中 的 元 素 按照 指定 的 格式 打印 了 出 来 ,关于 foreach 语句 的 具体 内 容 可 以 参见 第 8 章 的 条 
件 与 循环 。 
“Oils 
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The capital of SD is Jinan. 
| The capital of HEB is Shi jiazhuang. 
| The capital of HEN is Zhengzhou. 
The capital of HB is Wuhan. 
The capital of HN is Changsha. 
The capital of GD is Guangzhou. 
The capital of GX is Nanning. 


图 6-6 ”自动 遍历 数组 元 素 


通过 比较 手动 和 自动 遍历 数组 的 方法 可 知 ， 手 动 凯 有 历 适 合 于 小 规模 的 数组 遍历 ， 可 以 
引导 系统 指针 灵活 地 在 数组 元 素 间 来 回 移动 ， 而 自动 壳 历 适 合 于 大 规模 的 数组 遍历 ， 可 以 
引导 系统 指针 按照 一 定 顺序 逐一 地 在 数组 元 素 问 移 动 。 


6.2.2 ”如 何 给 数组 中 的 元 素 排序 


在 对 某 数组 排序 之 前 , PHP 是 按照 各 数组 元 素 被 创建 的 先后 顺序 来 排序 的 。 也 就 是 说 ， 
如 果 不 改 变数 组 中 各 元 素 的 顺序 而 直接 输出 数组 中 的 元 素 的 值 , 那么 PHP 处 理 引 擎 将 按照 
各 元 素 被 创建 的 先后 顺序 来 输出 各 元 素 的 索引 和 对 应 的 值 。 这 一 点 已 经 在 例 6.7 和 图 6-6 
所 示 中 得 到 了 印证 。 

在 编写 PHP 脚本 时 ， 可 以 按 索 引 或 元 素 值 对 数组 中 的 元 素 进行 排序 。 为 此 ，PHP 还 提 
供 了 一 些 用 于 数组 元 素 排序 的 函数 。 下 面 就 来 认识 一 下 它们 。 


1. sort($arrayName) 


该 函数 将 按照 指定 数组 中 各 元 素 的 值 的 大 小 对 数组 元 素 进行 排序 。 若 元 素 值 同 为 数 
字 ， 则 按照 数字 大 小 排序 ， 若 元 素 值 中 既 有 数字 ， 也 有 字符 串 ， 则 数字 排 在 字符 串 前 ， 字 
符 串 按照 大 小 写 和 字母 顺序 排序 。 同 一 字母 的 大 写字 母 排 在 小 写字 母 前 面 。 需要 注意 的 是 ， 
若 数 组 的 索引 为 字符 串 ， 如 例 6.7 所 示 中 定义 的 数组 $capitals， 在 使 用 sort0 函 数 后 ， 字 符 
串 索 引 将 全 部 转换 成 数字 。 

在 图 6-7 所 示 中 ， 可 以 看 到 ， 在 使 用 了 sort0 函 数 对 数组 $capitals 中 的 元 素 排序 之 后 ， 
第 一 个 创建 的 数组 元 素 Jinan 被 排 到 了 第 3 位 ， 而 第 五 个 创建 的 元 素 Changsha 则 被 排 到 了 
第 1 位 。Zhengzhou 则 因为 首 字母 是 Z 而 被 排 到 了 数组 的 最 后 一 位 。 同 时 ， 数 组 元 素 的 索 
引 也 由 之 前 字符 串 变 成 了 图 6-7 所 示 中 的 数字 。 

如 果 想 保留 数组 的 字符 串 形式 的 索引 ， 则 需要 使 用 asort0 方 法 。 


2. asort ($arrayName) 


该 函数 与 sort0 函 数 一 样 , 将 按照 指定 数组 中 各 元 素 值 的 大 小 对 数组 元 素 进行 排序 , 数 
字 在 前 ， 字 符 串 在 后 。 数 字 按 照 数字 大 小 排序 ， 字 符 串 按照 字母 顺序 和 大 小 写 排序 。 它 与 
sort0 函 数 唯一 不 同 的 就 是 ， 在 使 用 asort0 函 数 对 指定 数组 排序 后 ， 数 组 名 元 素 的 原 索 引 会 
予以 保留 。 也 就 是 说 图 6-7 中 被 数字 取代 的 数组 元 素 索引 会 变 成 如 图 6-8 所 示 的 样子 ， 数 
组 Scapitals 中 的 各 元 素 按照 元 素 值 的 大 小 排序 , 但 是 数组 元 素 的 原 索引 却 不 再 被 数字 取代 ， 
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而 是 得 到 了 保留 。 
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]=> 
string (8) “Changsha 
[1]=> 


1 
str: et) “Guangzhou” 
2]=: 

Dets) “JTinan” 

[3]=> 

string (T) “Nanming” 
[4]=> 

string(12) “Shijiazhuang” 


ie ©) “Wuhan” 
6]= 


Nip A “Thengzhou” 
图 6-7 使 用 sort0 函 数 的 效果 
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"SD"]=> 
string (5) "Jinan” 
[Gx"]=> 


string (7) “Nanning” 
["HEB”]=> 

string (12) "Shijiazhuang” 
["HB"]=> 

string() “When 
["HEN"]=:; 


string 名 “Zhengzhou” 
] 


图 6-8 使 用 asort0 函 数 的 效果 


与 sort0 和 asort0) 函 数 类 似 , PHP 还 提供 了 一 对 按照 指定 数组 中 各 元 素 值 从 大 到 小 进行 
排序 的 函数 : rsort0 和 arsort()。 其 中 rt 代表 reverse， 也 就 是 反 向 的 意思 。 我 们 车 使 用 rsortO 
和 arsort0 函 数 对 数组 Scapitals 中 的 元 素 进行 排序 ， 得 到 的 结果 将 如 图 6-9 所 示 。 在 图 6-9 
中 ， 左 边 为 使 用 rsort0 函 数 的 效果 ， 而 右边 则 为 使 用 arsort0 函 数 的 效果 。 

据 结 果 显 示 ，rsort0 将 指定 数组 中 各 元 素 的 值 按 照 首 字 母 在 字母 表 中 的 先后 顺序 从 后 
往 前 进行 了 排序 ， 于 是 之 前 在 图 6-7 中 排 末 位 的 Zhengzhou 排 在 了 第 1 的 位 置 ， 而 之 前 在 
图 6-7 中 排 首位 的 Changsha 则 排 到 了 末 位 。 与 图 6-7 所 示 类 似 的 是 ， 数 组 各 元 素 的 索引 按 
照排 序 的 结果 变 成 了 从 0 到 6 的 数字 了 。 而 arsort0 函 数 与 rsort0 函 数 一 样 ， 也 将 各 元 素 的 
值 按照 首 字母 在 字母 表 中 的 先后 顺序 从 后 往 前 进行 了 排序 。 但 是 ， 与 rsort0 函 数 不 同 的 是 ， 
arsort(O) 函 数 保留 了 指定 数组 各 元 素 原 有 的 字符 串 形 式 的 索引 。 

除 此 之 外 ，PHP 还 提供 了 另外 一 对 函数 : ksort0 和 krsort0。 其 中 , k 代表 key， 也 就 是 
说 这 一 对 函数 是 按照 数组 元 素 的 索引 值 对 数组 元 素 进行 排序 的 。 图 6-10 所 示 展 示 了 使 用 这 
两 个 函数 的 效果 。 

在 图 6-10 所 示 中 ， 左 边 为 使 用 ksortO 函 数 的 效果 ， 而 右边 为 使 用 krsortO 函 数 的 效果 。 
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在 使 用 ksort0) 函 数 对 数组 Scapitals 进行 排序 之 后 ， 排 在 第 一 位 的 为 Guangzhou， 
省 (GD) 的 首 字母 是 该 数组 中 所 有 省 份 名 称 的 首 字 母 在 字母 表 中 最 靠 前 的 一 个 ; 而 Jinan 
排 在 了 最 后 ， 则 是 因为 山东 省 (SD) 的 首 字 母 是 该 数组 中 所 有 省 份 名 称 的 首 字母 在 字母 表 


中 最 靠 后 的 一 个 。 


若 使 用 krsortO 函 数 对 数组 $capitals 中 的 元 素 进行 排序 ， 得 到 的 结果 则 与 使 用 ksortO 


相反 。 


string (9) “Zhengzhou” 
[1]=> 

string (5) “Wuhan” 
[2]=> 

string (12) “Shijiazhuang” 
[3]=> 

string (1) "Narning” 
[4]=> 

string (5) “Jinan” 

[5]=> 

string (9) "Guangzhou” 
[6]=> 

string (8) “Changsha” 


CE emp// ww greatwalltea chapter06/E6-07 01 php 


宽 Favorites | 荐 htpV/wwwgreatwallteaychapten6/E6-07.0LP 


array(7) { 
["HEN"]=> 
string (9) “Zhengzhou” 
[HB"]=> 

string (5) “Wuhan 
[HEB”]=> 

string (12) “Shijiazhuang” 
[Gx"]=> 

string (7) “Naming” 
[sp"]=> 

string (5) “Jinan” 
[6D"]=> 

string (9) “Guangzhou” 
CH]=> 

string (8) “Changsha” 


图 6-9 ”使 用 rsort0 和 arsort0 函 数 的 效果 


| Co Ee httpvwww greatwall tea/ chapter06/E6-07 01.php ~ 


宽 Favorites (Enttp//wwew greatwal ilteaychapter06/E6-07 .01.p-_- 国 


array(T) { 
["6D"]=> 
string (9) “Guangzhou” 
[Gx"]=> 
string (7) “Nanning” 
[HB"]=> 
string (5) “Wuhan” 
["HEB"]=> 
string (12) Shijiazhuang” 
["HEN"]=> 
st 
[HI 


string (5) “Jinan” 


2) http:/ /www.greatwallteal chapter06/Ex6-07 01,php 


全 ori 


string (5) “Jinan” 
[EN]=> 

string (8) “Changsha” 
[HEN"]=> 

string (9) “Zhengzhou” 
[HEB"]=> 

string (12) “Shijiazhuang” 
[HB"]=> 

string (5) “Wuhan” 
[Gx"]=> 

string (7) “Naming” 
[6D"]=> 

string (9) “Guangzhou” 


图 6-10 ”使 用 ksort0 和 krsort0 函 数 的 效果 
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表 6-1 对 这 三 组 六 个 函数 进行 一 下 总 结 。 
表 6-1 数组 排序 函数 


排序 函数 排序 效果 说 明 
sort($arrayName) 按 元 素 值 的 大 小 正 向 排序 ， 并 按 排序 结果 替换 元 素 的 索引 
asort($arrayName) 按 元 素 值 的 大 小 正 向 排序 ， 并 保留 元 素 原始 索引 
Tsort($arrayName) 按 元 素 值 的 大 小 反 向 排序 ， 并 按 排序 结果 替换 元 素 的 索引 
arsort($arrayName) 按 元 素 值 的 大 小 反 向 排序 ， 并 保留 元 素 原始 索引 
ksort($arrayName) 按 元 素 索 引 的 大 小 反 向 排序 
krsort($arrayName) 按 元 素 索引 的 大 小 反 向 排序 


在 这 一 节 中 ， 使 用 到 的 脚本 如 下 ， 供 大 家 参考 。 
【 例 6.8】 使 用 数组 排序 函数 为 数组 元 素 排序 。 


于 <?php 

所 $capitals = array("SD"” => "Jinan"， 

3 "HEB" => "Shijiazhuang", 
4 "HEN" => "Zhengzhou", 

5 "HB" => "Wuhan", 

6 "HN" => "Changsha", 

这 "GD" => "Guangzhou", 

8 "GX" => "Nanning"); 


10 sort ($capitals); // 按 元 素 值 的 大 小 正 向 排序 ， 并 按 排 序 结果 替换 元 素 的 索引 
加 #asort ($capitals); // 按 元 素 值 的 大 小 正 向 排序 ， 并 保留 元 素 原始 索引 
2 #rsort ($capitals); // 按 元 素 值 的 大 小 反 向 排序 ， 并 按 排序 结果 替换 元 素 的 索引 
3 #arsort ($capitals); // 按 元 素 值 的 大 小 反 向 排序 ， 并 保留 元 素 原 始 索引 


14 #ksort ($capitals); // 按 元 素 索引 的 大 小 反 向 排序 
TS #krsort ($capitals); // 按 元 素 索 引 的 大 小 反 向 排序 
16 echo "<pre>"; 

汪汪 var_dump ($capitals); 

18 echo "</pre>"; 

人 


大 家 可 以 移 除 第 10 行 一 第 15 行 相应 函数 前 的 注释 符号 〈#) 来 激活 该 函数 ， 或 在 相 
应 函数 前 添加 注释 符号 (#) 使 该 函数 失效 。 务 必 保证 一 次 只 激活 一 个 函数 。 若 多 条 函数 被 
同时 激活 ， 则 最 后 激活 的 函数 生效 。 


6.2.3 ”如何 比 较 数组 


有 时 候 ， 我 们 可 能 需要 比较 两 个 数组 中 的 元 素 是 否 相 同 。 这 时 可 以 使 用 amray_difftO 函 
数 ， 这 个 函数 可 以 带 两 个 或 两 个 以 上 的 参数 。 还 可 以 把 使 用 array_difftO 函 数 得 到 的 两 个 或 
多 个 数组 的 比较 结果 直接 赋值 给 一 个 新 的 数组 。 

【 例 6.9】 使 用 array_diftO 函 数 比 较 两 个 数组 中 的 元 素 是 否 相 同 。 
$arrayl = array("a" => "apple", "b" => "banana", "c" => orange ); 
$array2 = array ("grapefruit", "banana", "orange"); 
$diffArray = array diff(arrayl, array2); 
echo "<pre>"; 
var dump ($diffArray); 
echo "</pre>"; 
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在 如 例 6.9 所 示 的 脚本 中 ， 定 义 了 两 个 数组 : arrayl 和 array2。 其 中 ，arrayl 中 使 用 了 
字符 串 做 元 素 的 索引 , 而 array2 中 的 每 个 元 素 则 会 被 PHP 处 理 引 擎 自动 分 配 一 个 数字 索引 。 
接着 使 用 了 amray_difftO 函 数 比较 这 两 个 数组 的 不 同 ， 并 将 不 同 的 元 素 存 入 diffArray 数组 。 
最 后 使 用 var_dumpO 函 数 输出 数组 diffArray 的 结构 。 

在 运行 这 个 脚本 之 后 中 ， 可 以 发 现 ， 数 组 diffArray 只 有 一 个 元 素 : apple。 它 的 索引 为 
a， 因 为 这 个 元 素 只 存在 于 数组 arrayl 中 。 如 果 把 例 6.9 中 的 第 3 行 换 成 下 面 的 样子 : 


$diffArray = array diff (array2, arrayl1); 


那么 数组 diffArray 中 也 只 有 一 个 元 素 。 不 过 ， 这 次 不 是 apple， 而 是 grapefruit 了 。 它 
的 索引 也 不 是 a， 而 是 0 了 。 

从 这 个 示例 中 ， 可 以 知道 ，array_difftO 函 数 的 作用 是 找 出 第 一 个 数组 中 与 第 二 个 数组 
不 一 样 的 元 素 ， 比 较 结果 中 的 各 元 素 的 索引 保持 不 变 。 需 要 注意 的 是 ，array_difftO 函 数 会 
将 不 同 数组 中 值 相 同 的 元 素 看 成 是 相同 的 元 素 ， 无 所 谓 这 些 元 素 的 索引 是 否 相 同 ， 就 像 例 
6.9 中 的 banana 和 orange 一 样 。 

如 果 想 找 出 两 个 数组 中 元 素 索 引 或 元 素 值 不 一 样 的 所 有 元 素 ， 则 可 以 使 用 
array_diff assoc() 函 数 。 其 中 assoc 是 英文 单词 associate 的 缩写 ， 表示 关 联 性 索引 ， 也 就 是 
之 前 提 到 的 字符 串 索引 。 

现在 ， 把 例 6.9 中 的 第 3 行 换 成 下 面 的 样子 ， 看 看 运行 的 结果 有 什么 不 同 : 


$diffArray = array diff assocl(arrayl, array2); 


在 运行 了 修改 好 的 脚本 后 ， 数 组 diffArray 中 的 元 素 变 成 了 三 个 ， 它 们 分 别 是 索引 为 a 
的 apple、 索 引 为 b 的 banana 和 索引 为 c 的 orange。 因 为 数组 arrayl 和 array2 中 的 三 个 元 
素 要 么 索引 不 一 样 ， 要 么 元 素 值 不 一 样 ， 要 么 索引 和 值 都 不 一 样 。 

PHP 除了 提供 这 些 用 于 找 不 同 函数 之 外 ， 还 提供 了 另外 一 对 函数 : array_intersect() 和 
array_intersect_assoc()， 其 中 intersect 的 意思 是 交 又 。 这 两 个 函数 的 作用 是 找 出 两 个 或 多 个 
数组 的 交集 ， 前 者 只 要 求 元 素 值 相同 即 可 ， 而 后 者 则 要 求 元 素 索 引 和 元 素 值 都 相同 才 行 。 

现在 把 例 6.9 中 的 第 3 行 换 成 如 下 的 样子 : 


$simArray= array_intersect (arrayl, array2); 


然后 把 第 5 行 改 成 如 下 的 样子 : 

var_ dump ($simArray); 

这 时 候 ， 数 组 simArray 中 的 元 素 个 数 应 该 为 两 个 。 它 们 分 别 是 索引 为 b 的 banana 和 
索引 为 c 的 orange。 

如 果 把 例 6.9 中 的 第 3 行 换 成 如 下 的 样子 : 


$simArray= array interset assoc(arrayl, array2); 

那么 ，simArray 将 不 含 任何 元 素 ， 因 为 这 两 个 数组 中 的 每 个 元 素 要 么 索引 不 同 ， 要 么 
值 不 同 ， 要 么 索引 和 值 都 不 相同 。 

最 后 ， 我 们 还 像 上 一 节 一 样 对 这 两 组 四 个 函数 进行 一 下 总 结 ， 如 表 6-2 所 示 。 
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表 6-2 数组 比较 函数 
比较 函数 
array_diff($arrayl, $array2, ...) 
array_diff assoc($arrayl, $array2, ...) 
array_intersect($array1, $array2, ...) 


效果 说 明 
找 出 第 一 个 数组 中 与 第 二 个 数组 元 素 值 不 同 的 元 素 
找 出 第 一 个 数组 中 与 第 二 个 数组 元 素 值 或 索引 不 相同 的 元 素 
找 出 第 一 个 数组 中 与 第 二 个 数组 元 素 值 相同 的 元 素 
找 出 第 一 个 数组 中 与 第 二 个 数组 元 素 值 和 索引 均 相 同 的 元 素 


array_intersect_assoc($arrayl1, $array2, ...) 
6.3” 串 串门 一 一 数组 与 其 他 数据 类 型 的 互 转 


说 句 实话 ， 人 是 个 奇怪 的 动物 : 一 旦 可 以 完成 一 件 事 ， 就 会 想 着 怎么 把 这 件 事情 做 出 
花样 来 。 如 果 我 告诉 你 数组 可 以 被 转换 成 其 他 的 数据 类 型 。 虽 然 可 能 你 不 懂 如 何 转换 ， 但 
你 肯定 会 问 我 : 那 是 不 是 意味 着 其 他 的 数据 类 型 也 可 以 被 转换 成 数组 ? 

不 管 你 问 了 没 问 ， 这 都 不 重要 了 。 在 本 节 里 ， 我 们 就 来 讨论 一 下 为 什么 要 翻 来 倒 去 地 
折腾 着 把 数组 变 成 这 又 变 成 那 , 又 把 不 是 数组 的 变量 转换 成 数组 ,以 及 如 何 实现 这 种 转换 。 


6.3.1 为 什么 要 转换 


数组 ， 我 们 所 说 的 复杂 变量 ， 它 的 组 织 结构 说 紧密 也 谈 不 上 紧密 ， 因 为 想 要 往 里 添加 
些 元 素 或 者 从 中 间 删 除 些 元 素 看 上 去 是 那么 的 简单 ; 但 是 说 松散 也 不 见得 有 多 松散 ， 因 为 
想 把 它们 拆 开 ， 一 个 一 个 地 处 理 也 不 是 一 件 那么 简单 的 事情 。 说 白 了 ， 数 组 存储 的 就 是 一 
串 信 息 元 素 。 

有 时 候 ， 我 们 需要 把 数组 转换 成 一 个 字符 串 ， 而 这 个 字符 串 包 含 该 数组 的 所 有 的 元 素 
的 值 ， 有 时 候 ， 我 们 需要 把 有 着 特定 规律 〈 分 隔 符 ) 的 一 串 字 符 划 开 ， 把 这 串 字 符 中 的 某 
些 内 容 当 成 数组 的 元 素 ， 一 个 一 个 地 存储 起 来 。 只 要 确定 了 有 分 隔 符 ， 这 些 转换 对 于 PHP 
来 说 就 是 易如反掌 。 

其 实 ， 数 组 不 光 可 以 被 转换 成 字符 串 ， 也 不 是 只 有 字符 串 才 可 以 被 转换 成 数组 。 我 们 
可 以 把 任意 的 一 个 数组 拆 开 变 成 一 个 个 松散 独立 的 变量 ， 这 时 数组 中 的 元 素 索引 和 元 素 值 
分 别 变 成 了 独立 的 变量 名 和 变量 值 ; 我们 也 可 以 把 若干 的 变量 放 在 一 起 组 成 一 个 数组 ， 这 
独立 的 变量 名 和 变量 值 也 就 变 成 了 这 个 数组 中 各 元 素 的 元 素 索引 和 元 素 值 。 

那么 为 什么 要 这 人 么 干 呢 ? 

我 们 转换 来 、 转 换 去 的 理由 说 千 道 万 ， 总 结 起 来 具有 一 句 话 ， 那 就 是 “形势 需要 ”。 
例如 ， 可 以 将 用 户 的 姓名 、 性 别 、 年 龄 和 籍贯 之 类 的 基本 信息 存储 在 一 个 数组 中 ， 但 是 当 
需要 在 不 同 的 页 面 问 传递 这 些 数据 的 时 候 ， 可 以 选择 将 这 些 信息 整合 成 一 个 字符 串 ， 也 可 
以 选择 将 这 些 信息 按照 项 目的 不 同 转换 成 若干 个 字符 串 ， 用 于 信息 的 加 密 。 再 比如 ， 在 某 
个 页 面 接收 到 了 从 其 他 页 面 传递 过 来 的 经 过 转换 和 加 密 了 的 信息 ， 这 时 ， 就 需要 对 接收 到 
的 信息 进行 解密 和 重组 ， 以 便 还 原 成 可 以 处 理 的 数据 。 

可 能 你 对 于 上 述 的 情景 还 会 有 其 他 的 解决 办 法 ， 但 已 经 无 关乎 是 否 需要 转换 来 、 转 换 
去 这 件 事 本 身 了 。 在 下 面 两 小 节 的 内 容 中 ， 我 们 将 学 习 : 

口 如 何 拆 分 字符 串 成 数组 ， 又 如 何 整合 数组 元 素 成 字符 串 


“Ts 


口 如 何 拆 分 数组 成 变量 ， 又 如 何 整合 变量 成 数组 。 


6.3.2 ”数组 与 字符 串 的 互 转 


在 本 小 节 里 ， 我 们 用 数组 $capitals 来 学 习 如 何 整合 数组 元 素 成 字符 串 ， 以 及 如 何 拆 分 
字符 串 成 数组 。 
【 例 6.10】 拆 分 字符 串 成 数组 。 


5 <?php 

2 $capitals = array("SD" => "Jinan", 

， "HEB" => "Shijiazhuang", 
4 "HEN" => "Zhengzhou", 

5 "HB" => "Wuhan", 

6 "HN" => "Changsha", 

"GD" => "Guangzhou", 

8 "GX" => "Nanning"); 

六 

10 $stringIn = implode ("|", $capitals); 
了 echo $stringIn; 

E20 

在 这 段 脚本 中 ， 在 定义 了 数组 $capitals 之 后 ， 又 定义 了 一 个 变量 $stringm。 然 后 使 用 


implode() 函 数 为 变量 $stringIn 赋值 ， 最 后 输出 变量 $stringIn 的 值 。 其 中 implode() 函 数 可 以 
将 数组 中 各 元 素 的 值 整合 在 一 个 字符 串 中 。 
这 段 脚 本 输出 的 结果 如 下 : 


JinanlShijiazhuanglZhengzhoulWuhanlChangshalGuangzhoulNanning 


通过 观察 脚本 运行 的 结果 可 以 发 现 ， 数 组 Scapitals 中 各 元 素 的 值 被 “|” 隔 开 了 ， 而 “|” 
就 是 implode0 函 数 中 定义 的 分 隔 符 。 在 implode0 函 数 中 ， 可 以 使 用 任意 字符 串 做 分 隔 符 。 
接 下 来 ,来 看 看 如 何 把 变量 $stringIn 还 原 成 数组 。 
【 例 6.11】 拆 分 字符 串 成 数组 。 
$stringIn = " Jinan|Shijiazhuang|ZhengzhoulWuhanlChangshalGuangzhou 


INanning"; 
$arrayOut = explode("|", $stringIn); 


echo "<pre>"; 
Var dump ($arrayOut); 
echo "</pre>"; 


own 一 上 


在 这 段 脚 本 中 ， 定 义 了 一 个 新 的 数组 SarrayOut， 并 用 explode0 函 数 为 其 赋值 。 在 使 用 
explode(0) 函 数 时 ， 也 定义 了 一 个 分 隔 符 。 

这 段 脚 本 的 输出 的 内 容 如 图 6-11 所 示 。 

通过 截图 可 以 看 到 ， 变 量 $stringIn 的 值 被 拆 分 重组 成 了 一 个 拥有 七 个 元 素 的 数组 ， 这 
些 元 素 的 值 与 Scapitals 中 各 元 素 的 值 相同 ， 但 是 索引 却 不 一 样 。 原 来 使 用 explode0 函 数 生 
成 的 数组 ， 其 元 素 索 引 是 由 系统 按照 各 元 素 被 添加 进 数 组 的 先后 顺序 自动 分 配 的 。 这 一 点 
请 大 家 务必 要 注意 。 
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[cm me | 
ERROR 
突 Favorites | 大 htpy/wwwgreatwallteaychapten06/B6-1ILphp ， | 芥 ~ 7 加剧 Pagev safey Toosv 加- 
array(7) { 
| [0]=> 
string (6) " Jinan” 


[1]=> 


string (12) “Shijiazhuang” 


川 01=> 
string (9) 


string (7) 


“Thengzhou” 
“Wuhan” 
“Changsha” 
“Guangzhou” 


“Warming” 


图 6-11 拆 分 字符 串 成 数组 


6.3.3 ”数组 与 变量 的 互 转 


除了 将 指定 数组 中 的 所 有 的 元 素 整合 到 一 个 变量 中 ， 我 们 还 可 以 将 指定 数组 中 的 每 个 
元 素 变 成 一 个 独立 的 变量 。 这 样 一 来 ， 各 元 素 的 原始 索引 和 元 素 值 都 得 到 了 保留 。 

在 本 小 节 里 ， 还 是 继续 使 用 数组 Scapitals 来 举例 。 

【 例 6.12】 将 数组 拆 分 成 若干 个 独立 的 变量 。 


汪 <?php 

2 $capitals = array ("sD" => "Jinan", 

3 "HEB" => "Shijiazhuang", 
4 "HEN" => "Zhengzhou", 

5 "HB" => "Wuhan", 

6 "HN" => "Changsha", 

_ "GD" => "Guangzhou", 

8 "GX" => "Nanning"); 

9 

10 extract ($capitals); 

TL echo "The capital of Shandong is $SD.<br>"; 
12 echo "The capital of Henan is SHEN."7 


在 这 段 脚本 中 ， 定 义 了 含有 七 个 元 素 的 数组 Scapitals。 然 后 使 用 了 extract0 函 数 将 数组 
$capitals 拆 分 成 了 七 个 独立 的 变量 。 这 些 变量 的 变量 名 为 数组 元 素 的 原始 索引 ， 而 变量 值 
则 为 这 些 原 始 索引 对 应 的 元 素 值 。 接 下 来 使 用 echo 命令 输出 了 两 句 话 , 在 这 两 句 话 中 引用 
了 两 个 被 拆 分 出 来 的 变量 名 。 

这 段 脚 本 运行 的 结果 如 下 : 


The capital of Shandong is Jinan. 
The capital of Henan is Zhengzhou. 


我 们 可 以 在 这 段 脚本 的 末尾 加 上 如 下 的 内 容 来 把 这 些 个 被 拆 分 出 来 的 变量 再 整合 


3 $capitalsReunion = compact ("SD", "HEB", "HEN", "HB", "HN", "GD", "GX"); 
14 echo "<pre>"; 
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人 var dump ($capitalsReunion); 
16 echo "</pre>"; 
> 


在 脚本 的 第 13 行 ， 定 义 了 一 个 新 的 数组 $capitalsReunion， 然 后 使 用 compactO 函 数 为 
其 赋值 。 大 家 注意 ， 在 compact0 函 数 中 ， 变 量 名 是 以 字符 串 形式 书写 的 ， 前 面 没有 “3$” 
号 。 如 果 大 家 觉得 在 compactO 函 数 中 写 这 么 一 长 串 很 麻烦 的 话 ， 可 以 在 前 面 再 加 上 下 面 这 
一 句 : 

$arrayvIn = array("SD". "HBB “HEN HB "HM" "GO" GX) 

然后 把 第 13 行 改 成 : 


$capitalsReunion = compact ($arrayIn); 


这 里 ， 还 是 要 提醒 大 家 注意 一 下 ，compact0 函 数 中 引用 的 数组 必须 使 用 “$+ 数组 名 ” 
这 样 的 形式 ， 而 引用 的 变量 则 必须 使 用 字符 串 的 形式 ， 不 要 在 前 面 添加 “$” 号 。 

现在 ， 还 像 之 前 各 小 节 一 样 ， 总 结 一 下 在 这 一 小 节 里 学 到 的 两 组 四 个 函数 ， 如 表 6-3 
所 示 : 


表 6-3 数组 转换 函数 
效果 说 明 

使 用 指定 的 分 隔 符 将 指定 数组 中 各 元 素 的 值 组 合成 一 个 字符 串 
在 指定 的 字符 串 中 两 个 分 隔 符 之 问 的 内 容 做 为 一 个 元 素 添加 到 新 建 的 数 
组 中 
将 指定 数组 的 各 元 素 拆 分 成 独立 的 变量 ， 其 中 元 素 索 引 为 变量 名 ， 元 素 值 
为 变量 值 
将 指定 变量 组 合成 一 个 新 数组 , 使 用 这 些 变量 的 变量 名 为 新 数组 各 元 素 的 
元 素 索引 ， 这 些 变量 的 值 为 新 数组 各 元 素 的 值 


比较 函数 
implode(delimiter, $array) 


explode(delimiter, $string) 


extract($array) 


compact(varname, 
$arrayName, ...) 


最 后 看 看 这 段 脚本 输出 的 结果 ， 如 图 6-12 所 示 。 


ee9 -DIE 加 四 四 


用 宽 Favorites | 厦 htpV/www.greatwallteychapter06/B6-12.php 全 ~ ”加 基 > page Safetyv Toosv 力 ” 


The capital of Shandeng is Jinan. 
The capital of Henan is Zhengzhou. 


array(7) { 
["SD“]=> 
string (5) “Jinan” 
[HEB"]=> 
string (12) “Shijiazhuang” 
HEN*]=> 
“Thengzhou” 
“uhan” 


Changsha” 


@ Intemet | Protected Mode: of 给 ” 所 100% ~ 


图 6-12 数组 与 变量 的 互 换 
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6.4 分 分 合 合 一 一 数组 的 拆 分 与 合并 


天 下 巾 巾 ， 丝 为 利 来 ， 天 下 接 接 ， 皆 为 利 往 。 分 分 合 合 实 乃 最 常见 的 需求 了 ， 有 分 就 
有 合 ， 有 合 必 有 分 。 在 上 一 节 里 ， 我 们 学 习 了 如 何 把 数组 转换 成 一 个 或 多 个 变量 以 及 如 何 
把 一 个 变量 或 若干 个 变量 整合 成 数组 。 在 本 节 里 ， 我 们 将 学 习 更 加 灵活 的 数组 变换 方式 : 
数组 的 拆 分 和 合 


6.4.1 如 何 拆 分 数组 


不 知 大 家 是 和 否 还 记得 在 第 5 章 的 第 5.3.1 小 节 的 文字 游戏 中 学 习 过 一 招 叫 * 断章取义 ”。 
说 的 是 用 substrO 函 数 截取 一 个 字符 串 中 的 某 一 部 分 生成 一 个 新 的 字符 串 。 它 的 格式 如 下 : 


substr ($stringName, start pos,substr len) 


其 中 ，$stringName 代表 一 个 指定 的 字符 串 变量 名 ，start_pos 代表 子 字符 串 在 母子 字符 
串 中 的 起 始 位 置 ， 而 substr_len 代表 子 字符 串 的 长 度 。 

拆 分 数组 使 用 到 的 函数 与 substr0 函 数 类 似 ， 这 个 函数 叫 array_slice0，slice 就 是 切片 
的 意思 。 它 的 格式 如 下 : 


array_ slice($arrayName, offset, subArray len, preserve keys); 


其 中 ，$arrayName 代表 一 个 指定 的 数组 名 ; offset 指 的 是 子 数组 首 个 元 素 相对 于 母 数 
组 首 个 元 素 的 偏离 值 ， 也 就 是 子 数组 的 起 始 位 置 ，subArray_len 指 的 是 子 数组 包含 的 元 素 
个 数 ， 而 preserve_keys 是 个 Boolean 类 型 的 参数 ， 如 果 需 要 为 子 数组 各 元 素 中 保留 其 在 母 
数组 中 的 原始 索引 ， 可 将 preserve_keys 设 为 “TRUE”， 否 则 将 其 设 为 “FALSE”。 

需要 注意 的 是 ，preserve keys 这 个 变量 的 取 值 默认 为 “TRUE” ， 也 就 是 说 在 没有 特 
殊 需 要 的 时 候 ， 可 忽略 这 个 变量 的 设置 。 

在 本 小 节 里 ， 还 是 以 数组 $capitals 为 例 来 看 看 如 何 从 这 个 数组 中 拆 分 出 一 个 子 数 组 。 

【 例 6.13】 拆 分 数组 。 


人 <?php 
区 Scapitals = array("SD"” => "Jinan", 

各 "HEB" => "Shijiazhuang", 
4 "HEN" => "Zhengzhou"， 

5 "HB" => "Wuhan", 

6 "HN" => "Changsha", 

汉 "GD" => "Guangzhou"， 

8 "GX" => "Nanning"); 


及 

0 $subArray = array slice($capitals, 1, 5); 
4 echo "<pre>"; 

了 Var dump ($subArray); 

3 echo "</pre>"; 

3 


在 这 一 段 脚 本 中 ,定义 了 一 个 拥有 七 个 元 素 的 数组 Scapitals 以 及 一 个 新 数组 $subArray， 
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然后 使 用 array_sliceO 函 数 截取 了 从 数组 Scapitals 第 二 位 开始 一 共 5 个 元 素 并 将 它们 加 入 到 
数组 SsubArray 中 。 最 后 使 用 var_dump0 〇 函数 输出 了 这 个 子 数 组 的 结构 ， 结 果 如 图 6-13 
所 示 。 


| 王 区 7 | 

[|@e - Er 
宽 Favorites 独 http://www.greatwalltea/chapter06/Ex6-13.php | 从 7 可 基 Pagev safeyv Toosv 右 - 
array (5) { 


["HEB"]=> 

string (12) “Shijiazhuang” 
["HEN”]= 

string (9) “Zhengzhou” 
["HB"]=> 

string (5) “Wuhan” 
[HN"]=> 

string (8) “Changsha” 
["6D"]=> 

string (9) “Guangzhou” 


图 6-13 拆 分 数组 


在 图 6-13 所 示 中 ， 我 们 可 以 发 现 子 数组 $subArray 一 共 拥 有 五 个 元 素 ， 第 一 个 元 素 为 
母 数组 的 第 二 个 元 素 ， 最 后 一 个 元 素 为 母 数组 的 第 六 个 元 素 。 而 母 数组 中 的 第 一 个 元 素 和 
最 后 一 个 元 素 都 被 截 掉 了 。 


6.4.2 ”如 何 合并 数组 


就 像 在 本 节 开 头 说 的 那样 : 天 下 之 事 ， 有 合 就 有 分 ， 有 分 必 有 合 。 那 么 在 这 一 小 节 里 ， 
就 看 看 如 何 合并 两 个 或 多 个 数组 。 

为 了 解决 合并 数组 的 问题 ，PHP 也 给 我 们 提供 了 一 个 好 用 的 函数 ， 那 就 是 
array_merge()， 其 中 ，merge 意思 就 是 合并 。 它 的 格式 如 下 : 


array merge ($arrayNamel, $arrayName2, ..); 


这 个 函数 的 变量 要 求 非常 的 简单 ， 直 接 将 需要 合并 的 数组 罗列 起 来 就 可 以 了 。 但 是 凡 
事 都 有 两 面 ， 看 似 简单 的 东西 ， 实 则 不 一 定 简单 。 这 个 array_mergeO 函 数 就 是 这 样 : 虽然 
结构 简单 ， 但 运行 的 结果 却 有 时 让 人 摸 不 着 头脑 。 我 们 一 起 来 看 一 段 脚 本 。 

【 例 6.14】 合并 数组 。 


时 <?php 
$arrayl = array("a", "b" => "b"); 
$array2 = array("A", "b" => "B"); 


3 

4 

5 SmergedArray = array merge ($arrayl, $array2); 
6 echo "<pre>"; 

7 var_ dump ($mergedArray); 

8 echo "</pre>"; 

9 > 


在 脚本 的 开始 ， 定 义 了 两 个 数组 $arrayl 和 $array2。 数 组 $arrayl 包含 两 个 元 素 ， 其 中 
第 二 个 元 素 的 索引 为 “b”， 值 为 “b”; 数组 $array2 也 包含 两 个 元 素 ， 其 中 第 二 个 元 素 的 


二 


索引 为 “b”， 值 为 “B”。 接 着 我 们 使 用 array_mergeO 函 数 将 这 两 个 数组 合并 在 了 一 起 并 
将 合并 后 的 数组 赋值 给 了 $mergedArray， 最 后 我 们 使 用 var dump0) 函 数 输 出 了 数组 
$mergedArray 的 结构 。 大 家 可 以 推测 一 下 这 个 脚本 的 执行 结果 ， 然 后 再 看 看 图 6-14 所 示 ， 
检查 一 下 与 你 推测 的 结果 是 否 相同 呢 ? 


E http://www.greatwalltea/ chapter06/E6-14.php 人 租 ~ 国 "本 对 page Safetyv Toosv 恩 - 


| string (1) “B” 
[1]=> 

| string (1) “aA” 

} 


图 6-14 合并 数组 


看 到 这 个 结果 ， 也 许 有 的 同学 会 欢呼 ， 有 的 同学 要 叹气 了 。 合 并 后 的 数组 一 共 只 有 三 
个 元 素 ， 其 中 数组 $arrayl 和 $array2 中 的 第 一 个 元 素 都 得 到 了 保留 ， 而 只 有 数组 $array2 中 
的 第 二 个 元 素 得 到 了 保留 。 这 是 为 什么 呢 ? 

PHP 处 理 引擎 在 执行 这 段 脚本 的 时 候 ， 会 轮流 比 对 两 个 数组 的 每 个 元 素 。 当 它 发 现 数 
组 $arrayl 和 $array2 中 的 第 一 个 元 素 的 索引 相同 时 且 均 为 数字 索引 时 ， 它 就 会 再 为 在 新 数 
组 中 为 数组 $array2 的 第 一 个 元 素 重 新 编号 并 将 其 置 于 新 数组 的 末尾 。 当 它 发 现 数组 $arrayl 
的 第 二 元 素 的 索引 为 字符 串 且 与 数组 Sarray2 的 第 二 元 素 的 索引 相同 时 , 它 就 会 直接 用 数组 
$array2 的 第 二 元 素 的 值 覆 盖 数 组 $arrayl 的 第 二 元 素 的 值 并 保留 这 个 相同 的 索引 。 

简单 地 来 说 ， 相 同 数字 索引 的 元 素 会 被 重新 索引 ， 使 得 这 些 元 素 的 值 均 得 以 保留 ， 而 
对 于 相同 字符 串 索 引 的 元 素 ， 这 些 元 素 值 只 有 一 个 会 被 保留 下 来 且 使 用 其 原始 索引 。 

现在 把 array_mergeO 函 数 中 两 个 数组 的 位 置换 一 下 ， 结 果 又 是 什么 呢 ? 大 家 可 以 再 进 
行 推测 一 番 

最 后 ， 简 单 总 结 一 下 这 一 节 中 学 到 的 两 个 函数 如 表 6-4 所 示 : 

表 6-4 拆 分 与 合并 数组 
拆 分 与 合并 函数 效果 说 明 

通过 指定 数组 , 子 数组 首 元 素 相 对 母 数组 首 元 素 的 偏 
离 值 和 子 数组 中 元 素 的 个 数 来 截取 母 数组 中 的 若干 
元 素 生 成 子 数组 
通过 指定 若干 数组 ， 生 成 不 含 重复 元 素 值 的 合并 数组 


array_slice($arrayName, offset, subArray_len) 


array_merge($arrayNamel, $arrayName2, ...) 


6.5 多 维 数 组 


看 到 “多 维 ”这 个 词 ， 大 家 可 能 就 会 想 ， 难 不 成 我 们 之 前 讨论 的 数组 还 只 是 数组 的 一 
种 吗 ? 是 的 。 之前, 我 们 讨论 的 是 数组 中 的 一 维 数 组 。 说 白 了 ， 就 是 只 有 一 层 索引 的 数组 。 


人 


这 里 的 多 维 指 的 是 有 多 层 索引 的 数组 。 

在 PHP 脚本 中 ， 我 们 可 以 定义 数组 索引 层 数 是 没有 限制 的 。 不 过 ， 如 果 定 义 的 数组 的 
索引 超过 了 两 层 ， 就 得 考虑 修改 脚本 了 。 简 单 地 说 ， 在 编写 PHP 脚本 时 ， 如 果 需 要 使 用 多 
维 数组 的 话 ， 最 多 只 用 三 维 数组 就 好 。 


6.5.1 多 维 数组 vs. 一 维 数组 


为 了 说 清楚 “多 维 数组 ”和 “一 维 数组 ”之 间 的 区 别 ， 我 们 先 来 看 一 张 超市 小 票 ， 如 
图 6-15 所 示 。 


Purchased Price 
Sainsburys 
屿 Sulhing wn ta Apples 1.29 
Pears 1.69 
Po 00 Mem Asparagus 2.19 
ASPARAGUS (PERU, 6312 NILES) 
ns i 7 Grapes 1.85 
STRAWBEARIES (SPRIN，958 NILES) 2.79 
| Lettuce 0.87 
POTATOES 《1SRREL ，2187 WILES) 2.49 
| | Tomroes matemrim, 6865 ILES) 1.35 Strawberries 2.79 
CARROTS (1SRAEL ，2187 NILES) 2.29 
ORENENTINGS (00L1VIN, G250 NILES) £719 Broccoli 1.06 
13 LTEMS, BALANCE OUE 28.47| Spinach 1.47 
er Potatoes 2.49 
引 3 Bs 
| an 加 Tomatoes 1.35 
2 
Ef Carrots 2.29 
Ps 
四 和 Peas 1.34 
a 3s 3 
Clementines 2.79 


图 6-15 超市 小 票 
这 张 小 票 中 反映 出 来 的 信息 如 果 用 一 个 一 维 数 组 来 表示 ， 就 是 如 下 的 样子 : 


$purchased[apples] = 1.29; 
$purchased[pears] = 1.69; 
S$purchased[asparagus] = 2.19; 
S$purchased[grapes] = 1.85; 
S$purchased[lettuce] = 0.87; 
S$purchased[strawberries] = 2.79; 
$purchased[broccoli] = 1.06; 


通过 这 个 一 维 数组 ， 我 们 可 能 很 方便 地 查找 和 输出 某 件 商 品 的 价格 。 但 是 如 果 这 张 小 
票 上 的 商品 有 数 百 多 种 ， 这 种 查找 和 输出 是 十 分 消耗 服务 器 资源 的 。 为 了 避免 这 种 情况 ， 
我 们 再 来 看 看 这 张 小 票 ， 仔 细 找 找 其 中 的 内 在 规律 。 
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心细 的 同学 也 许 已 经 发 现 了 ， 这 张 小 票 上 不 是 水 果 ， 就 是 蔬菜 。 那 么 可 以 把 它们 分 成 
两 类 ， 然 后 建立 一 个 二 维 数组 ， 也 就 是 如 下 的 样子 : 


Spurchased[fruit] [apples] = 1.29; 
Spurchased[fruit] [pears] = 1.69; 
$purchased[vegetable] [asparagus] = 2.19; 
$purchased[fruit] [grapes] = 1.85; 
$purchased[vegetable] [lettuce] = 0.87; 
$purchased[fruit] [strawberries] Zl 
$purchased[vegetable] [broccolil] 432306 


我 们 也 可 以 把 这 个 二 维 数组 看 成 是 由 数组 组 成 的 数组 , 它 的 结构 可 以 被 看 做 是 如 表 6-5 
所 示 的 样子 : 
表 6-5 ”二 维 数组 的 结构 


Value 
Key 
Key Value 
apples 1.29 
pears 1.69 
fruit grapes 1.85 
Purchased strawberries 279 
clementines 2.79 
asparagus 2.19 
lettuce 0.87 
vegetable - 
broccoli 1.06 
spinach 1.47 


通过 这 张 表 ， 可 以 清楚 地 看 到 这 个 二 维 数组 Spurchased 是 由 两 个 一 维 数组 $fruit 和 
$vegetable 组 成 的 ， 每 个 一 维 数组 又 含有 若干 个 元 素 。 


6.5.2 ”创建 多 维 数组 和 查看 数组 结构 


在 上 一 小 节 中 , 通过 分 析 一 张 超市 小 票 ， 理解 了 二 维 数组 和 一 维 数组 的 区 别 。 那 就 是 ， 
二 维 数组 可 以 被 看 做 是 由 多 个 一 维 数组 组 成 的 。 简 单 来 说 ， 多 维 数组 和 一 维 数组 的 区 别 就 
在 于 前 者 是 以 数组 为 元 素 的 ， 而 后 者 是 以 变量 为 元 素 的 。 

创建 一 个 多 维 数组 ， 既 可 以 像 上 一 节 里 那样 ， 一 个 一 个 地 定义 数组 中 的 每 个 变量 ， 也 
可 以 使 用 arrayO 函 数 ， 就 像 创建 一 维 数组 一 样 。 

如 果 用 array0 函 数 来 创建 一 个 二 维 数组 用 于 反映 图 6-15 所 示 的 这 张 超市 小 票 的 内 容 的 
话 ， 脚 本 应 该 是 如 下 的 样子 : 


Spurchased = array ("fruit" => array("apple" => 1.29, pears => 1.69, ..), 

"vegetable" => array("asparagus" => 2.19, lettuce => 0.87, ..)); 

在 这 段 脚本 中 , 使 用 了 三 次 array0 函 数 ,第 一 次 使 用 array0 函 数 创建 了 数组 Spurchased， 
定义 了 两 个 元 素 fruit 和 vegetable, 第 二 次 和 第 三 次 使 用 array0 函 数 分 别 将 fruit 和 vegetable 
定义 为 数组 并 向 两 个 数组 中 添加 了 若干 元 素 。 

如 果 我 们 用 var_dump0 函 数 输出 数组 Spurchased 的 结构 ， 就 应 该 是 如 图 6-16 所 示 的 


i 
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样子 。 


http/ /www.gr Se | 


”~ 
次 Fworites | 态 http//www.greatwalltea/chapter06/E6-15.php | | 租 " 国 " 口 时 page safey Toosv 加 


array(2) { 

[fruit"]=> 

array(5) { 
["apples”]=> 
float (1. 29) 
[pears”]=> 
float (1. 69) 
[grapes”]=> 
float (1. 85) 
[strawberries ]=> 
float (2.79) 
["element ines“]=> 
float (2. 79) 

} 

["vegetable”]=> 

array(8) { 
[asparagus”]=> 
float (2. 19) 
["lettuce”]=> 
float (0. 87) 
[“broccoli”]=> 
float (1. 06) 
[spinach”]=> 
float (1.47) 
[potatoes“]=> 
float (2.49) 
["tonatoes”]=> 


© Intermet | Protected Mode: Of 


图 6-16 输出 二 维 数组 的 结构 


输出 这 个 二 维 数组 结构 的 原始 脚本 如 下 。 
【 例 6.15】 输出 二 维 数组 的 结构 。 


1 <?php 

2 S$purchased = array ("fruit™" => array( "apples" => 1.29, 
3 "pears" => 1.69, 

4 "grapes" => 1.85, 

号 "strawberries" => 2.79, 

6 "clementines" => 2.79), 

六 "vegetable" => array("asparagus" => 2.19, 
8 "1Lettuce" => 0.87, 

9 "broccoli" => 1.06, 


10 "spinach" ee 
2 "potatoes" => 2.49, 
12 "tomatoes" => 1.35, 
23 "carrots" => 2.29, 
14 "peas" => 1.34)); 
15 

16 //display the structure of S$purchased 

全 有 echo "<pre>"; 

18 var_dump ($Spurchased) 

Es, echo "</pre>"; 

OMe 


6.5.3 ”如 何人 遍历 多 维 数 组 


遍历 多 维 数组 与 遍历 一 维 数组 的 方法 是 类 似 的 。 我 们 可 以 使 用 在 6.2.1 小 节 中 讲 到 的 


" 
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foreach 


语句 来 遍历 多 维 数组 。 值 得 注意 的 是 ，foreach 语句 是 支持 嵌 套 使 用 的 。 也 就 是 说 ， 


一 个 数组 有 几 层 ， 我 们 就 得 嵌 套 儿 层 foreach 语句 。 
还 记得 foreach 语句 的 结构 吗 ? 还 是 先 来 回顾 一 下 : 
foreach ($arrayName as $key => $value); 
如 果 需 要 榜 套 使 用 的 话 ， 就 变 成 了 下 面 的 样子 : 


foreach ($arrayName as SupperLayer ) 


上 
} 


foreach( $upperLayer as $lowerLayerKey => LowerLayerValue) 
{ 
// 在 这 里 可 以 对 数组 元 素 进行 一 些 操作 


现在 我 们 用 foreach 语句 来 输出 上 一 小 节 中 定义 的 二 维 数组 Spurchase， 脚 本 如 下 。 
【 例 6.16】 遍历 二 维 数组 。 


<?php 

$purchased = array ("fruit" => array( "apples" => 1.29, 
paars” => 1.697 
"grapes" => 1.85, 
"strawberries" => 2.79, 
"clementines" => 2.79), 

"vegetable" => array("asparagus" => 2.19, 

"lettuce" => 0.87, 
"broccoli" => 1.06, 
"spinach" => 1.47, 
"potatoes" => 2.49, 
"tomatoes" => 1.35, 
"carrots" => 2.29, 
"peas" => 1.34)); 


// 遍 历数 组 Spurchased 
foreach ($purchased as $category){ 
foreach ($category as $food => $price) { 
$f price = sprintf("%01.2f", $price); 
echo "$food: \$$f price <br>"; 


在 foreach 语句 中 ，PHP 处 理 引 擎 是 如 何 动作 的 呢 ? 我 们 一 起 来 看 一 看 : 


口 


口 口 口 


PHP 处 理 引擎 首先 读 取 数组 Spurchased 的 第 一 个 元 素 〈 键 值 对 ) ， 然 后 把 这 个 元 
素 的 值 存 入 变量 $category 中 。 需 要 注意 的 是 这 个 数组 Spurchased 的 第 一 个 元 素 是 
一 个 数组 ， 于 是 变量 $category 变 成 了 数组 $category。 (第 17 行 ) 

然后 PHP 处 理 引 擎 读 取 数组 Scategory 的 第 一 个 元 素 〈 键 值 对 ) ， 然 后 把 这 个 元 素 
的 索引 存 入 变量 $food 中 ， 把 其 对 应 的 值 存 入 变量 Sprice 中 。 (第 18 行 ) 

接着 PHP 处 理 引擎 按照 事先 规定 好 的 格式 将 价格 格式 化 。 (第 19 行 ) 

紧 接着 使 用 echo 语句 输出 一 行 记录 。 (第 20 行 ) 

当 内 府 的 foreach 语句 在 数组 Scategory 中 找 不 到 可 以 读 取 的 内 容 时 ， 内 内 foreach 
语句 循环 结束 ,返回 到 上 一 层 的 foreach 语句 。 这 时 PHP 处 理 引擎 会 读 取 $purchased 
的 第 二 个 元 素 〈 键 值 对 ) 并 将 这 个 元 素 的 值 存 入 数组 Scategory 中 。 (第 17 行 》 
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口 然后 PHP 处 理 引擎 读 取 数 组 $category 中 的 第 一 个 元 素 〈 键 值 对 ) ， 然 后 把 这 个 元 
素 的 索引 存 入 变量 $food 中 ， 把 其 对 应 的 值 存 入 变量 Sprice 中 。 (第 18 行 ) 
口 然后 按照 事先 规定 好 的 格式 格式 化 价格 ， 并 输出 记录 直至 结束 。 (第 19、20 行 ) 
口 当 外 部 foreach 语句 找 不 到 可 以 读 取 的 内 容 时 ， 循 环 结束 。 (第 22 行 ) 
简单 地 来 说 ， 外 部 的 foreach 语句 将 读 取 到 的 第 一 个 元 素 $fruit 的 值 存 入 $category 中 ， 
然后 内 嵌 的 foreach 语句 开始 遍历 $category。 当 数组 $category 遍历 完成 时 ， 外 部 的 foreach 
语句 开始 读 取 第 二 个 元 素 $Svegetable， 并 将 读 取 到 的 值 再 次 存 入 $category 中 。 接 着 内 柑 的 
foreach 语句 开始 遍历 $category， 直 至 循环 结束 。 


6.6 ”实战 练习 : 级 联 下 拉 菜 单 


6.6.1 界面 预览 


在 本 小 节 里 我 们 利用 所 学 的 知识 实现 如 图 6-17 所 示 的 级 联 下 拉 菜 单 。 当 在 第 一 个 下 拉 
菜单 中 选择 了 一 个 值 时 ， 第 二 个 下 拉 菜 单 里 的 内 容 会 随 之 发 生变 化 。 


全 Fworte | 着 二 了 下 芝 杠 


级 联 下 拉 杠 


= 请 过 择 直 夺 市 == 忆 |[== 请 运 拓 而 秋 区 = 


北京 市 海淀 区 


图 6-17 级 联 下 拉 框 


在 这 个 练习 中 ， 需 要 重点 关注 一 下 数组 的 拆 分 以 及 数组 元 素 的 遍历 。 例 子 的 实现 过 程 
中 涉及 到 HTML DOM 和 JavaScript 的 相关 知识 ， 已 通过 注释 进行 了 解释 。 如 果 还 有 不 明 
白 的 地 方 ， 可 以 在 网 上 查找 相关 资料 。 


6.6.2 ”实现 过 程 
首先 ， 我 们 还 是 来 看 看 HTML 部 分 的 代码 : 


"I 
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<!DOCTYPE html> 


<html> 
<head> 
<meta charset="UTF-8"> 
<title> 级 联 下 拉 框 </title> 
<style> 
body { 
font-size:small; 
font-family:'Microsoft YaHei'; 
} 
form { 
text-align:center; 
F 
hot 
text-align:center; 
下 
#info { 
font-size:xx-large; 
line-height:60px; 
text-align:center; 
i 
</style> 
</head> 
<body> 
<h3> 级 联 下 拉 框 </h3> 
<hr /> 
<form action='?check' method="post"> 
<select name="prov" id="prov"></select> 
<select name="city" id="city"></select> 
<button type="submit"> 提 交 </button> 
</form> 
<div id="info"></div> // 在 用 户 提交 了 表单 后 ， 展 示 用 户 选 择 的 结果 
/* 在 下 面 这 段 JavaScript 脚本 中 ， 我 们 使 用 了 getElementById 将 两 个 下 拉 列 表 
框 分 别 赋值 给 名 为 prov 和 city 的 对 象 ， 然 后 通过 该 对 象 的 add 方法 向 对 象 中 添加 
列表 项 。 在 后 续 的 PHP 脚本 中 ， 我 们 也 反复 使 用 到 了 这 段 JavaScript 脚本 */ 
<script language="JavaScript"> 
Var prov = document .getElementById ("prov"); 
prov.options.add (new Option ("== 请 选择 直辖 市 ==", '=")); 
var city = document .getElementById("city"); 
city.options.add (new Option ("== 请 选择 市 辖区 ==", '="')); 
</script> 
</body> 
</html> 


在 HTML 部 分 中 ， 除 了 那 段 JavaScript 脚本 之 外 ， 与 我 们 在 之 前 的 章节 中 看 到 的 内 容 
没有 什么 不 同 。 现 在 再 来 看 看 PHP 的 部 分 : 
<?php 


Smulicipality = explode (','，,' 北 京 , 天津, 上 海 , 重庆 '); 
$beijing = explode ('，'，' 东 城区 ,西城 区 ,海淀 区 , 朝阳 区 ,丰台 区 ,石景山 区 ,通州 区 ， 
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“20 


顺义 区 ,房山 区 , 大兴 区 ,昌平 区 ,怀柔 区 ,平谷 区 , 门头沟 区 ,密云 县 ,延庆 县 '); 

Stianjin = explode (',',' 和 平 区 ,河西 区 ,南开 区 ,河东 区 ,河北 区 , 红 桥 区 , 东 丽 区 , 津 
南 区 , 西 青 区 ,北辰 区 , 滨海 新 区 , 武 清 区 , 宝 坛 区 , 曹县, 宁 河 县 ,静海 县 ' ) ; 

$shanghai = explode ('，'，' 黄 浦 区 ,徐汇 区 ,长 宁 区 ,静安 区 ,普陀 区 , 闸北 区 ,虹口 区 ， 
杨浦 区 , 闵行 区 ,宝山 区 ,嘉定 区 , 浦东 新 区 , 金山 区 ,松江 , 青浦 区 ,奉贤 区 ,崇明 县 ") ; 
$chongqing = explode(',', ' 渝 中 区 , 大 渡口 区 ,江北 区 , 沙坪 坝 区 , 九龙 坡 区 , 南岸 区 ， 
北碚 区 , 綦江 区 , 双 桥 区 , 渝 北 区 , 巴 南 区 , 万 州 区 , 涪陵 区 , 黔江 区 ,长 寿 区 ,江津 区 ,合川 区 , 永 
川 区 ,南川 区 '); 


/* 在 上 面 的 五 句 中 ， 我 们 使 用 explode () 函数 将 五 串 由 逗号 分 隔 的 字符 串 转换 成 了 数组 。 
其 中 ， 第 一 个 数组 Smulicipality 中 的 内 容 将 填充 到 如 图 6-17 所 示 的 左边 的 下 拉 框 中 ， 
而 余下 四 个 数组 则 会 根据 用 户 在 左边 下 拉 框 中 的 选择 ， 填 充 到 右边 的 下 拉 框 中 。 */ 


$bjSstr 
$tjstr 
$shstr 
$cqstr 


// 上 面 这 四 个 变量 是 用 来 存放 处 理 成 vavaScript 脚本 的 数组 元 素 


foreach ($mulicipality as $key => $value) { 
echo '<script>prov.options.add (new Option("'.$value.'","'.$key. 
a 六 


} 
// 上 面 这 段 脚 本 用 来 遍历 数组 $mulicipality， 并 将 其 键 和 值 拼合 到 JavaScript 脚本 中 


foreach ($beijing as $key => $value) { 
$bjStr .= 'city.options.add(new Option("'.$value.'","'.S$key. 
人 

} 


foreach ($tianjin as $key => $value) { 
$tjSstr .= "city.options .add (new Option("'.$value.'","'.S$key. 
ee 


| 


foreach ($shanghai as S$key => S$value) { 
$shstr .= 'city.options.add(new Option("'.$value.'", 
"Skey.™)) e's 

} 


foreach ($chongqing as $key => $value) { 
$cqstr .= 'city.options.add(new Option("'.$value.'", 
"Skey.'")) 2's 

} 


// 上 面 这 四 段 脚本 用 来 遍历 剩余 的 四 个 数组 ， 并 将 它们 的 键 和 值 拼接 到 对 应 的 JavaScript 
脚本 中 


echo '<script language="JavaScript"> // 从 此 往 下 为 Javascript 脚本 
Prov.onchange = function () { 
// 当 prov 下 拉 列 表 框 的 值 发 生 改变 时 ， 会 触发 如 下 函数 


selectedProv = prov.options[prov.selectedIndex] .value; 


// 获 取 用 户 的 选择 
switch(selectedProv){ 
Case Os 
removeAll (); // 移 除 当前 city 下拉 列表 框 中 的 所 有 列表 项 


city-options .add (new Option ("== 请 选择 市 辖区 ==", "=") );'. 
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// 向 city 下 拉 框 中 添加 列表 头 
$bjSstr. // 用 PHP 向 city 下 拉 框 中 添加 北京 市 的 市 辖 
'break; 

CaSG. 1s 
removeAll () 7 
city.options.add (new Option ("== 请 选择 市 辖区 ==", "="));'. 
StjStr- // 用 PHP 向 city 下 拉 框 中 添加 天 津 市 的 市 辖 
'break; 

Cel; :1 hd 
removeAll (); 
city.options.add (new Option ("== 请 选择 市 辖区 ==", "="));'. 
$shstr. // 用 PHP 向 city 下 拉 框 中 添加 上 海 市 的 市 辖 
'break; 

CASD IMs 
removeAll (); 
city.options.add (new Option ("== 请 选择 市 辖区 ==", "=") );' 
$cqstr. // 用 PHP 向 city 下 拉 框 中 添加 重庆 市 的 市 辖区 


'break; 


区 


Xx. 


x, 


} 


function removeAll() { 
for (i=0;i<20;i++){ 
city.options.remove (city.options[i]); 
// 这 里 使 用 了 city 对 象 的 remove 方法 移 除 指定 的 对 象 
} 
} 
x/ script> 


/* 拼合 JavaSscript 脚本 ， 并 输出 。 在 上 面 这 段 脚本 中 ， 我 们 定义 了 在 如 图 6-7 所 示 的 页 
面 中 ， 当 用 户 在 左边 的 下 拉 框 中 选择 了 某 个 值 时 ， 浏 览 器 会 记录 用 户 选 择 的 列表 项 对 应 的 
值 ， 然 后 根据 获取 到 的 值 向 右边 的 列表 框 中 添加 列表 项 */ 


if(isset($ REQUEST['check'])){ // 判 断 用 户 是 否 提交 了 表单 


$prov = $ REQUEST['prov']; // 获 取 用 户 选择 的 直辖 市 
Scity = $ REQUEST['city']; // 获 取 用 户 选择 的 市 辖区 
if(Sprov == "=" || $city == '="){ // 判 断 用户 是 否 做 出 了 选择 
$msg = ' 请 选择 直辖 市 和 市 辖区 '; 
} else { 
switch ($prov) { // 将 用 户 的 选择 转换 成 可 读 的 文字 
Cas "Oa 
$msg = ' 北 京 市 ' .$beijing[intval ($city)]; 
break; 
case '1': 
Smsg = ' 天 津 市 ' .$tianjin[intval ($city)]; 
break; 
cae "2 
$msg = "上 海 市 ' .$shanghai [intval ($city)]; 
break; 
Ce 
Smsg = ' 重 庆 市 ' .$chongqing[intval ($city)]; 
break; 


} 
} 
// 向 ID 为 info 的 DIV 输出 用 户 的 选择 


人 
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echo "<script language="javascript">document. 
getElementById("info") .innerHTML="" .$msg.'"</script>"'; 

上 面 这 段 脚 本 一 共 完 成 实现 了 如 下 两 个 功能 : 

(1) 在 用 户 选择 了 一 个 直辖 市 后 ， 其 右边 的 市 辖区 的 内 容 也 会 发 生变 化 。 

(2) 在 用 户 提交 所 选 城市 和 市 区 后 ， 在 表单 的 下 方 会 以 粗 体 展示 用 户 所 选 。 

下 面 来 看 看 上 面 的 代码 是 如 何 实现 这 两 个 功能 的 。 

功能 1 在 用 户 选择 了 一 个 直辖 市 后 ， 其 右边 的 市 辖区 的 内 容 也 会 发 生变 化 。 

首先 我 们 定义 了 五 个 变量 ， 分 别 为 Smunicipality 、$beijing 、$tianjin 、$shanghai 和 
$chongqing 用 来 存储 这 四 个 城市 和 每 个 城市 所 辖 地 区 的 名 称 。 这 些 变量 都 是 使 用 explode0 
函数 生成 的 数组 变量 。 

接 下 来 ， 定 义 了 四 个 空 变量 ， 用 来 存放 JavaScript 语句 。 需 要 注意 的 是 ， 在 一 个 PHP 
文件 中 ， 只 有 PHP 脚本 和 非 PHP 脚本 之 分 ， 我们 可 以 在 任何 非 PHP 脚本 中 通过 “<?php” 
和 “?>” 来 插入 PHP 脚本 。 以 $bjStr 为 例 ， 我 们 使 用 foreach 语句 对 $beijing 这 个 数组 进行 
遍历 ， 并 将 遍历 的 值 混合 到 JavaScript 的 数组 语句 中 ， 如 下 所 示 : 

$bjStr .= 'city.options.add(new Option('".$value."','".$key."'));'; 


在 这 条 语句 中 ， 我 们 使 用 “.” 操 作 符 连接 了 city.options.addnew Option('".$value. 
"$key." ')) 三 个 字符 串 以 及 $value 和 S$key 两 个 PHP 变量 的 值 。 其 中 ，“city.options.add” 
中 的 “city* 是 我 们 在 HTML 中 定义 了 一 个 下 拉 框 。 另 一 个 下 拉 框 为 “prov”; “options” 
是 下 拉 框 元 素 的 子 元 素 ; 而 “add” 则 是 向 下 拉 框 中 添加 子 元 素 的 方法 。 这 些 都 是 JavaScript 
语句 ， 在 这 里 就 不 展开 了 。 

在 定义 好 $bjStr、$tjStr、$shStr 和 $cqStr 之 后 ,我 们 就 开始 向 HTML 页 面 中 添加 JavaScript 
代码 。 在 这 里 定义 了 “prov” 下 拉 框 的 onchange0) 函 数 ， 当 “prov” 下 拉 框 的 值 发 生变 化 时 ， 
就 会 加 载 相应 城市 的 所 辖区 。 代 码 如 下 : 


echo '<script language="JavaScript"> // 从 此 往 下 为 vavaScript 脚本 
Prov.onchange = function () { 
// 当 prov 下 拉 列 表 框 的 值 发 生 改变 时 ， 会 触发 如 下 函数 
selectedProv = prov.options[prov.selectedIndex] .value; 
// 获 取 用 户 的 选择 
switch(selectedProv){ 
case "0": 
removeAll (); // 移 除 当前 city 下 拉 列 表 框 中 的 所 有 列表 项 
city.options.add (new Option ("== 请 选择 市 辖区 ==", "="));'. 
// 向 city 下 拉 框 中 添加 列表 头 
$bjstr. // 用 PHP 向 city 下 拉 框 中 添加 北京 市 的 市 辖区 
'break; 
Case. "in 
removeAll (); 
city.options.add (new Option ("== 请 选择 市 辖区 ==", "="));'. 
StjStr // 用 PHP 向 city 下 拉 框 中 添加 天 津 市 的 市 辖区 
'break; 
CaS Ns 
removeAll (); 
city.options.add (new Option ("== 请 选择 市 辖区 ==", "="));'. 
$shstr. // 用 PHP 向 city 下 拉 框 中 添加 上 海 市 的 市 辖区 
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'break; 
Case "3": 
removeAll (); 
city.options.add (new Option ("== 请 选择 市 辖区 ==", "=") );"'. 
$cqstr. // 用 PHP 向 city 下 拉 框 中 添加 重庆 市 的 市 辖区 


'break; 
} 


function removeAll() { 
for (i=0;i<20;i++){ 
city.options.remove (city.options[i]); 
. // 这 里 使 用 了 city 对 象 的 remove 方法 移 除 指定 的 对 象 
人 
在 这 段 向 HTML 页 面 中 添加 内 容 的 JavaScript 代码 中 ， 我 们 同样 是 使 用 了 “.” 操 作 符 
连接 了 JavaScript 代码 和 PHP 变量 。 值 得 注意 的 是 ， 在 这 段 JavaScript 代码 中 ， 还 定义 了 
-个 removeAl0 函 数 ， 用 来 在 加 载 用 户 所 选 城市 辖区 的 同时 ， 移 除 之 前 加 载 的 城市 辖区 。 
这 样 ， 我 们 就 实现 了 第 一 个 功能 。 
功能 2 在 用 户 提交 所 选 城市 和 市 区 后 ， 在 表单 的 下 方 会 以 粗 体 展示 用 户 所 选 。 
为 了 实现 这 个 功能 ， 我 们 做 了 两 件 事 情 。 第 一 件 事 ， 就 是 在 HTML 页 面 的 表单 中 定义 
了 “action” 和 “method” 两 个 属性 。 其 中 “action” 属 性 值 为 “?check”， 而 “method” 
的 值 为 “post”。 前 者 定义 了 处 理 用 户 提交 数据 的 页 面 ， 而 后 者 则 定义 了 向 该 页 面 传递 用 
户 提交 数据 的 方式 。 关 于 如 何在 页 面 间 传 递 数 据 ， 大 家 可 以 参考 9.1 节 的 内 容 。 
这 里 大 家 只 要 知道 ， 我 们 将 用 户 提交 的 数据 传递 到 了 当前 页 面 ， 并 通过 如 下 的 脚本 来 
判断 和 显示 用 户 提交 的 内 容 就 可 以 了 。 
if (isset ($ REQUEST['check'])){ // 判 断 用 户 是 否 提交 了 表单 
Sprov = $ REQUEST['prov']; // 获 取 用 户 选择 的 直辖 市 


Scity = $ REQUEST['city']; // 获 取 用 户 选择 的 市 辖区 
if($prov == "=' || $city == '="){ // 判 断 用 户 是 否 做 出 了 选择 
$msg = ' 请 选择 直辖 市 和 市 辖区 '; 
} else { 
switch ($prov) { // 将 用 户 的 选择 转换 成 可 读 的 文字 
Case '0': 
$msg = ' 北 京 市 ' .$beijing[intval ($city)]; 
break; 
case '1': 
$msg = ' 天 津 市 ' .$tianjin[intval ($city)]; 
break; 
Case. “25 
Smsg = "上海 市 ' .$shanghai [intval ($city)]; 
break; 
Case '3': 
$msg = ' 重 庆 市 ' .$chongqing[intval ($city)]; 
break; 
lL 
} 


// 向 ID 为 info 的 DIV 输出 用 户 的 选择 
echo "<script language="javascript">document .getElementById 
("info") .innerHTML="" .$msg.'"</script>"'; 


ss 
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); 
二 
在 这 段 脚 本 里 ， 我 们 使 “isset($_ REQUEST['check])” 来 判断 用 户 是 否 提交 了 表单 。 这 
里 的 “$_REQUST[]” 是 个 预定 义 数组 ， 用 来 存放 用 户 通过 URL 或 表单 传递 的 数据 。 如 果 
“isset($_REQUEST['check"])” 的 结果 为 真 ， 则 表示 用 户 提交 了 表单 ; 反之 , 则 没有 提交 表单 。 
如 果 判 断 当 前 用 户 提交 了 表单 ， 则 获取 用 户 通 过 表单 提交 的 直辖 市 和 市 辖区 ， 并 将 这 
两 个 信息 存 入 变量 $prov 和 $city 中 。 由 于 我 们 通过 JavaScript 生成 的 下 拉 框 中 的 值 和 名 分 
别 为 之 前 定义 的 Sminucipality、$beijing、Stianjin、$shanghai 和 $chongqing 这 五 个 数组 各 元 
素 的 索引 和 值 ， 所 以 用 户 通过 表单 传递 过 来 的 只 是 所 选 直 辖 市 和 市 辖区 的 索引 值 ， 因 此 我 
们 需要 根据 索引 值 ， 再 回 到 上 述 五 个 数组 中 查找 对 应 的 直辖 市 和 市 辖区 的 名 字 。 另 外 ， 需 
要 注意 的 是 , 用户 通 过 表单 传递 过 来 的 值 , 无 论 其 是 数字 还 是 字符 ， 其 数据 类 型 都 是 字符 ， 
羽 此 ， 我 们 使 用 了 “intval0” 这 个 函数 转换 了 一 下 数据 类 型 ， 以 便 在 数组 中 查找 用 户 通 过 
表单 传递 过 来 的 索引 对 应 的 城市 和 市 区 名 称 。 
将 查 到 了 用 户 通过 表单 传递 过 来 的 索引 对 应 的 城市 和 市 区 名 称 后 ， 我 们 使 用 一 段 
JavaScript 代码 将 获取 到 的 内 容 和 输出 到 HIML 页 面 中 定义 的 ID 为 “info” 的 DIV 元 素 中 。 
这 样 ， 我 们 就 实现 了 第 二 个 功能 。 


6.7 习 题 
(1) 请 判断 下 面 脚本 的 运行 结果 : 
<?php 
$a = array("a" => 5, "b" => "banana", "c" => "egg"); 


$b = array(10, "apple","pie"); 
= array merge ($a, $b); 
$d = array slice($a,0,2); 
echo "<pre>";var dump ($c);echo "</pre>"; 
echo "<pre>";var dump($d);echo "</pre>"; 


(2) 请 使 用 手动 遍历 的 方法 输出 上 面 脚本 中 数组 $a 的 所 有 元 素 。 

(3) 请 定义 一 个 包含 10 个 元 素 的 数组 ， 然 后 使 用 foreach 语句 遍历 这 个 数组 。 
(4) 请 使 用 implodeO 函 数 合 并 你 定义 的 数组 为 一 个 字符 串 。 

(5) 请 使 用 explode0 函 数 将 上 一 题 中 的 字符 串 转换 成 一 个 数组 。 
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在 之 前 的 章节 中 ， 除 了 第 6 章 使 用 了 foreach 语句 的 脚本 之 外 ， 其 他 的 脚本 都 是 直线 
型 的 〈 不 包括 实战 练习 中 的 脚本 示例 ) 。PHP 处 理 引 擎 会 按照 从 上 到 下 的 顺序 执行 脚本 中 
的 每 一 行 。 如 果 想 要 调整 某 脚 本 中 语句 的 执行 顺序 ， 直 接 将 需要 提前 执行 的 语句 放 到 前 面 
就 可 以 了 。 

如 果 世 界 总 是 如 此 的 简单 ， 那 该 有 多 美好 啊 ， 可 惜 这 种 美好 只 存在 于 理想 世界 中 。 在 
现实 世界 里 ， 我 们 会 有 着 无 穷 无 尽 的 目标 需要 去 达成 ， 只 有 满足 了 某 些 条 件 ， 我 们 才能 实 
现 这 些 目标 。 比 如 ， 如 果 想 要 涨 工资 ， 就 得 好 好 干 活 ， 只 有 业绩 达标 了 ， 才 有 可 能 给 你 涨 
工资 。 再 比如 ， 如 果 想 要 拿 奖 学 金 ， 就 得 埋头 若干 地 好 好 学 习 ， 只 有 学 习 成 绩 上 去 了 ， 才 
有 希望 拿 奖 学 金 。 另 一 方面 ， 要 想 业 绩 达标 ， 就 得 日 复 一 日 地 重复 着 手头 的 工作 ， 要 想 提 
升学 习 成 绩 ， 就 得 年 复 一 年 地 读书 考试 。 所 谓 的 做 一 天 和 尚 撞 一 天 钟 ， 讲 的 就 是 这 个 道理 。 

几乎 所 有 的 计算 机 语言 都 是 为 了 让 计算 机 程序 蔡 人 们 做 些 简单 的 判断 以 及 模拟 人 的 
某 些 重复 性 的 活动 ， 从 而 把 人 从 繁重 的 劳动 中 解放 出 来 。 这 听 上 去 有 些 像 是 在 讲 马 克 思 主 
义 政治 经 济 学 了 。 那 我 们 言 归 正 传 ， 看 看 PHP 是 如 何 通 过 脚本 模拟 这 些 活动 的 吧 。 


7.1 精细 化 运算 一 一 条 件 


我 们 还 是 用 在 第 3 章 里 提 到 的 一 个 例子 来 说 说 什么 是 条 件 。 这 个 例子 是 这 样 的 : 
if (it is morning) 

get up; 

brush my teeth; 

wash my face; 


put on my jacket; 


1 
2 
3 
4 
5 
6 
yl go to work; 
8 


} 

在 这 个 例子 中 ， 第 1 行 定义 了 一 个 条 件 。 只 有 当 这 个 条 件 得 到 满足 时 ， 才 会 执行 第 3 
到 第 7 行 的 内 容 。 换 句 话 说， 只 有 当前 时 间 是 早晨 时 ， 我 们 才 会 进行 诸如 起 床 、 刷 牙 、 洗 
脸 、 穿 衣 、 赶 着 去 上 班 这 些 活 动 。 如 果 当 前 时 间 是 下 午 ， 我 们 就 有 另外 一 套 动作 了 。 

再 来 看 一 个 例子 。 不 知道 大 家 有 没有 注意 到 ， 当 我 们 成 功 登 录 了 腾讯 QQ 邮箱 后 ， 会 
在 邮箱 首页 上 看 到 如 图 7-1 所 示 的 欢迎 矢 ， 这 个 欢迎 辞 还 是 会 随 着 时 间 发 生变 化 的 : 早上 
登录 的 时 候 ， 它 会 说 “早上 好 ”， 中 午 登录 的 时 候 ， 它 会 说 “中 午 好 ”。 晚 上 登录 的 时 候 ， 
它 会 说 “晚上 好 ”。 

实现 这 个 功能 其 实 非常 简单 。 我 们 可 以 通过 定义 一 些 条 件 ， 让 电脑 输出 不 同 的 信息 。 

【 例 7.1】 定义 条 件 。 
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<?php 
$currentHour = date("H"); 
if ($currentHour < 12) { 
echo "Good morning!"; 
} elseif ($currentHour > 12 AND $currentHour <18){ 
echo "Good afternoon!™"; 
} elseif ($currentHour >18 AND $currentHour < 24){ 
echo "Good evening!"; 
$ 
a 
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图 7-1 QQ 邮箱 首页 


例 7.1 中 的 脚本 模拟 了 QQ 邮箱 首页 的 问候 功能 。 首 先 ， 将 当前 时 间 的 小 时 数 赋值 给 


变量 $currentHour， 然 后 使 用 条 件 语 句 定义 了 如 下 三 个 条 件 : 


口 $currentHour 小 于 12; 

口 $currentHour 大 于 12 日 小 于 18; 

口 $currentHour 大 于 18 日 小 于 24。 

通过 比 对 当前 时 间 与 这 些 条 件 的 匹配 情况 来 要 求 PHP 处 理 引擎 输出 不 同 的 问候 语 。 在 


yA | 


脚本 中 , 第 3 行 、 第 5 行 和 第 7 行 括 号 里 的 内 容 就 是 用 PHP 语言 对 这 三 个 条 件 的 描述 。 下 
面 就 具体 来 看 看 如 何 定义 条 件 。 


什么 是 条 件 


在 定义 条 件 之 前 ， 需 要 知道 什么 是 条 件 。 
在 PHP 脚本 中 ， 条件 是 一 种 表达 式 ， 通常 用 在 复杂 脚本 中 。 它 的 判断 结果 通常 只 有 两 


人 
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种 ， 一 种 为 “true”， 另 一 种 为 “false”。PHP 引擎 根据 条 件 的 判断 结果 来 决定 脚本 中 某 
些 部 分 是 否 需 要 执行 。 定 义 一 个 条 件 ， 需 要 比较 两 个 或 多 个 值 的 大 小 。 在 比较 值 的 大 小 时 
通常 会 问 到 的 问题 包括: 
口 两 个 值 是 否 相 等 ? 比如， 你 喝 的 啤酒 的 品牌 和 我 喝 的 是 不 是 一 样 的 ? 再 比如 ， 你 
的 名 字 是 不 是 叫 金三顺 ? 
口 两 个 值 中 ， 是 不 是 一 个 比 另 一 个 大 ? 比如 ， 你 的 年 纪 是 不 是 比 我 大 ? 再 比如 ， 你 
买 的 这 套 衣服 得 1000 多 块 钱 吧 ? 
口 某 个 字符 串 的 格式 是 否 与 指定 的 样式 匹配 ?比如 ， 你 是 不 是 姓 李 ? 你 们 家 是 不 是 

住 在 郡 阳 街 ? 
当然 ， 我 们 可 以 一 次 定义 多 个 条 件 ， 并 指定 这 多 个 条 件 之 前 的 关系 。 比 如 ， 你 爸爸 是 
不 是 在 红旗 厂 工 作 ， 要 不 就 是 在 晨光 厂 工 作 ? 再 比如 ， 你 哥哥 今年 15 岁 ， 你 妹妹 今年 也 
12 岁 了 吧 ? 前 面 两 个 问题 之 间 的 关系 是 “或 ”， 而 后 面 两 个 问题 之 间 的 关系 是 “和 ”。 这 
也 是 定义 多 个 条 件 时 最 常用 的 两 种 关系 。 

总 结 一 下 ， 所 谓 条 件 ， 其 实 就 是 简单 疑问 句 ， 也 就 是 可 以 用 是 或 否 来 回答 的 问 句 。 在 
PHP 中 使 用 条 件 ， 只 需要 把 问题 用 脚本 语言 写 出 来 就 成 了 。 那 么 应 该 怎么 写 呢 ? 

在 写 之 前 ， 还 是 先 来 掌握 一 些 定义 条 件 时 常用 的 “词汇 ”和 “ 句 式 ” 吧 。 


7.1.2 ”如何 定 义 条 件 


在 本 小 节 里 ， 我 们 就 按照 上 一 小 节 中 提 到 的 三 类 问题 来 讲 一 讲 如 何 定义 条 件 吧 。 
1. 值 的 比较 
第 一 类 和 第 二 类 问题 讲 的 是 值 的 比较 。 在 比较 的 时 候 ， 会 使 用 一 些 比较 操作 符 来 连接 
被 比较 的 两 个 对 象 ， 如 表 7-1 所 示 。 
表 7-1 常见 的 比较 操作 符 


操 作 符 含义 
= 操作 符 前 后 的 两 个 变量 值 是 否 相等 
一 一 操作 符 前 后 的 两 个 变量 的 类 型 和 值 是 否 都 相等 
> 操作 符 前 的 变量 值 是 否 比 操作 符 后 的 变量 值 大 
>= 操作 符 前 的 变量 值 是 否 大 于 等 于 操作 符 后 的 变量 值 
< 操作 符 前 的 变量 值 是 否 比 操作 符 后 的 变量 值 小 
< 操作 符 前 的 变量 值 是 否 小 于 等 于 操作 符 后 的 变量 值 
{~ 操作 符 前 后 的 两 个 变量 的 值 是 否 不 相等 
! 一 操作 符 前 后 的 两 个 变量 的 类 型 和 值 是 否 都 不 相等 


这 些 操作 符 可 以 连接 的 不 仅仅 是 数字 , 也 可 以 用 于 连接 字符 串 。 有 的 同学 可 能 不 理解 ， 
字符 串 怎 么 能 比较 大 小 呢 ? 在 PHP 中 ,字符 串 的 大 小 是 有 规定 的 。 按 照 字母 表 的 顺序 ， 大 
写字 母 在 前 ， 小 写字 母 在 后 。 举 个 例子 “SOS” 就 要 比 “So” 大 。 标 点 符号 之 前 也 可 以 比 
较 大 小 ， 只 不 过 ， 我 们 通常 不 会 这 么 干 。 因 为 比较 标点 符号 的 大 小 没有 什么 实际 意义 。 

我 们 来 看 一 些 例子 : 
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$a = $b 

$age != 12 

SageU > $ageMe 

$suitePrice > 1000 

在 上 面 的 四 个 例子 中 ， 第 一 个 例子 比较 特别 ， 在 比较 两 个 变量 的 值 是 否 相 等 的 时 候 ， 
用 的 是 两 个 等 号 (一) ， 而 不 是 一 个 等 号 (=) 。 一 个 等 号 在 PHP 脚本 里 叫 赋值 符 ， 比 如 
“$a=16” 的 意思 是 把 16 这 个 值 赋 给 变量 $a, 而 两 个 等 号 才 是 比较 操作 符 , 比如 “$a 一 $b” 
的 意思 是 变量 $a 和 变量 $b 的 值 是 否 相等 。 两 者 的 区 别 务 必要 牢记 ， 不 要 用 错 了 。 

PHP 处 理 引擎 在 看 到 这 些 条 件 后 ,会 对 操作 符 前 后 的 两 个 变量 按照 操作 符 的 意义 进行 
比较 。 如 果 比 较 的 结果 与 操作 符 的 定义 相符 ， 返 回 一 个 布尔 值 tue。 如 果 比 较 的 结果 与 操 
作 符 的 定义 不 相符 ， 则 返回 另 一 个 布尔 值 false。 

比如 , 假设 $a= 1, 而 $b=2。 那么 当 PHP 处 理 引 擎 处 理 条 件 $a 一 $b 时, 会 返回 false， 
因为 两 个 变量 的 值 不 相等 。 

除了 使 用 操作 符 之 外 ， 我 们 还 可 以 使 用 一 些 判 断 函 数 来 进行 判断 。 

PHP 提供 了 许多 的 判断 函数 ， 比 较 常见 的 有 如 下 几 种 ， 如 表 7-2 所 示 。 


表 7-2 判断 函数 


判断 函数 含 义 


ry 否 已 定义 。 若 含有 多 个 对 象 ， 则 其 中 一 个 对 
isset($varNamel,$varName2,... ) ed ee 有 多 个 对 象 ， 则 其 人 对象 


is_int($varName) 判断 指定 对 象 是 否 为 整 型 变量 
is_array($varName) 判断 指定 对 象 是 否 为 数组 

is float($varName) 判断 指定 对 象 是 否 为 浮 点 型 变量 
is_string($varName) 判断 指定 对 象 是 否 为 字符 串 型 变量 

is_ numeric($varName) 判断 指定 对 象 是 否 为 数字 或 数字 型 字符 串 


is_null($varName) 判断 指定 对 象 的 值 是 否 为 堆 


这 些 判 断 函数 可 能 帮助 我 们 很 快 地 对 一 个 或 多 个 对 象 的 类 型 进行 判断 ， 从 而 对 这 些 对 
象 做 出 相应 的 动作 。 我 们 也 可 以 在 这 些 判断 函数 前 面 加 上 “!” 来 做 出 相反 的 判断 。 比 如 
“lisset0” 可 以 用 来 判断 指定 的 一 个 或 多 个 对 象 是 否 未 定义 ，“lis_stringO0” 则 可 以 用 来 判 
断 指 定 对 象 是 否 不 是 一 个 字符 串 。 


2. 正则 表达 式 


在 上 一 小 节 中 关于 第 三 类 条 件 讨论 的 其 实 就 是 模式 匹配 的 问题 。 为 了 实现 模式 匹配 ， 
就 得 先 定 义 一 个 模式 ， 而 定义 模式 就 要 用 到 正则 表达 式 。 接 下 来 ， 简 单 地 讨论 一 下 正则 表 
大 家 如 果 用 过 MS-DOS 操作 系统 ， 那 么 一 定 用 过 类 似 于 下 面 的 命令 : 


C:\ dir ex*.txt 


在 MS-DOS 操作 系统 中 ， 这 条 命令 可 以 打印 出 C 盘 根 目录 下 所 有 的 以 “ex” 开 头 的 
TXT 文件 。 这 个 “*” 叫 通配符 。 正 则 表达 式 与 通配符 类 似 ， 但 是 前 者 稍微 要 复杂 一 些 。 

最 常 使 用 正则 表达 式 的 地 方 就 是 检测 用 户 输入 的 内 容 是 否 符合 要 求 。 比 如 在 中 国 ， 手 
机 号 码 都 是 以 1 开头 的 ， 当 声称 是 来 自 中 国 的 用 户 输入 了 一 串 不 是 以 1 开头 的 数字 就 不 能 
被 存 和 数据库。 再 比如 湖北 省 的 身份 证 号 码 都 是 以 42 打头 的 , 当 声 称 是 来 自 湖北 省 的 用 户 
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输入 一 串 不 是 以 和 2 开头 的 字符 串 就 也 不 能 被 存 入 数据 库 。 
正则 表达 式 中 除了 有 英文 字符 之 外 ， 还 有 一 些 特殊 字符 。 先 来 看 看 有 哪些 特殊 字符 ? 
以 及 这 些 特殊 字符 的 作用 ， 如 表 7-3 所 示 。 

表 7-3 正则 表达 式 中 的 特殊 字符 


特殊 字符 含义 举例 匹配 字 串 不 匹配 字 串 
以 …… 开 始 egress ingress 
$ 以 …… 结 束 ham Hen 
任 一 字 简 up, great A,1 
a 前 一 位 字符 可 选 gem, germ gm 
() 必 售 字 锌 sg(enm germ gem, grem 
[] 可 选 字符 gl[erlm gem, grm germ, gel 
约 不 可 选 字符 g['erlm gym, gum 和 Sm， grem, 
在 前 置 和 后 置 字符 | 、 
- 之 间 的 所 有 字符 [A-Jleep Jeep, Beep Keep 
A i Shop1， 
上 项 > 5 
前 置 项 目 可 多 选 Shop[1-3]+ Shopl2 Shop321 Shop456 
前 置 项 目 可 重复 ge*m gm, geeem germ 
前 置 项 目 必 重 复 次 
{n} 数 ge{S}m geeeeem geem, geeem 
前 置 项 目 可 重复 次 geem, geeem, 
i 3 
{nl,n2} 数 范围 ge{2,5}m geeeem, geeeeem em 
后 置 项 目 不 含 特殊 | 、、 本 
入 含义 g\*m g*m ggm, gem 
(|) 可 选 字 串 I want an (apple | | Iwant an apple. ee 
orange). I want an orange. 


表 里 的 内 容 看 上 去 是 不 是 很 复杂 呢 ? 下 面 有 几 道 选择 题 ， 大 家 可 以 尝试 解答 一 下 ， 消 
化 消化 上 面 讲 到 的 内 容 。 

1.“ 一 串 以 任意 字母 开头 任意 长 度 的 字符 串 ” 可 用 正则 表达 式 表 示 为 : 

a) [A-Za-z].* b) ^[A-Za-z].*c) ^[A-Za-z] * d) ^[A-Za-z]. 

在 这 一 题 中 ， 需 要 定义 一 串 以 任意 字母 开头 且 长 度 不 定 的 字符 串 ， 可 以 用 到 的 特殊 符 
号 应 该 有 : 
口 “^” 标 记 起 始 位 置 ， 
口 “[-]” 包 含 从 A 到 Z 和 a-z 的 所 有 字母 ; 
口 “.*” 标 记 任意 长 度 的 字符 串 。 
这 样 一 分 析 后 ， 答 案 就 呼之欲出 了 ， 应 该 是 b。 
2. 下 面 哪 一 个 正则 表达 式 可 以 匹配 “My Dear Kim” 这 个 字符 串 ? 
a) “Dear (Kim | Jim) b) Dear [Kim]$ 
c) Dear (Kim | Jim) d) Dear {Kim,Jim} 
口 在 这 一 题 中 的 各 个 选项 中 ， 除 了 选项 a 用 了 “^” 标 示 起 始 字符 为 “D” 以 外 ， 其 
他 各 项 都 没有 用 起 始 位 置 符 。 而 我 们 需要 匹配 的 字符 串 的 起 始 字符 为 M。 选 项 a 
排除 。 
口 选项 b 中 使 用 了 “[]” 和 “$” 符 标志 可 与 其 匹配 的 字符 串 只 能 是 “… Dear K”、 
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“…Deari” 和 “…Dearm” 这 三 个 字符 串 。 选 项 b 排除 。 
口 选项 c 使 用 了 “(|)” 标 示 二 选 一 , 与 之 匹配 的 字符 串 可 以 是 “… Dear Kim” 和 “… 
Dear Jim”。 于 是 My Dear Kim 这 个 字符 串 是 可 以 匹配 该 正则 表达 式 的 字符 串 之 一 。 
选项 c 正确 。 
口 选项 d 使 用 了 “{,}” 标 示范 围 ,但 用 错 了 参数 。 这 个 特殊 符号 之 间 的 参数 必须 为 

数字 。 选 项 d 排除 。 

3. 下 面 哪 一 个 正则 表达 式 可 以 匹配 “9001” 和 “7851-32564” 这 样 的 两 串 字符 ? 

a) ^[0-9] {4}\-[0-9] {5}$ b) ^[0-9]{4}(\-[0-9] {5})$ 

c) ~^[0-9]{4}(\-[0-9]{5})2$ qd) “~[0-9] {4}\-[0-9]{5}?$ 

“9001” 和 “7851-32564” 这 样 的 两 串 字 符 ， 前 者 是 由 四 个 数字 组 成 的 ， 可 以 写成 
“人 ^[0-9]{4}$ ”。 后 者 是 由 四 个 数字 加 个 连 字 符 再 加 五 个 数字 组 成 的 ， 可 以 写成 
人 ^[0-9]{4}\-[0-9]{5}$， 也 就 是 选项 a。 所 以 说 选项 a 只 能 匹配 “7851-32564” 这 样 的 字 串 ， 
却 不 能 匹配 “9001” 这 样 的 字 串 。 如 果 想 要 让 正则 表达 式 “^[0-9]{4}\-[0-9]{5}$” 也 匹配 
“9001” 这 样 的 字 串 ， 我 们 就 得 对 其 进行 改造 。 

再 来 找 找 这 两 个 字 串 的 共同 点 : “9001” 是 由 四 个 数字 组 成 ， 而 “7851-32564” 是 由 
四 个 数字 开头 的 ， 也 就 是 说 把 连 字 符 加 后 面 的 五 个 字符 变 成 可 选 的 就 行 了 。 选 项 b 和 选项 
d 都 有 标示 前 置 字符 为 可 选 的 “? ”。 两 者 的 区 别 就 在 于 选项 b 中 用 了 一 对 “( )” 把 连 字 
符 和 五 个 数字 组 成 的 字 串 标示 为 必 含 字 符 了 。 这 样 一 来 ， 选 项 的 中 “? ”的 作用 范围 就 
由 五 个 数字 变 成 了 连 字 符 加 五 个 数字 了 。 而 选项 d 的 中 “? ”的 作用 范围 只 是 五 个 数字 而 
已 。 至 于 选项 c 中 的 “()” 不 过 是 强调 这 个 字符 串 必 须 以 连 字符 加 五 个 数字 结尾 器 了 。 

4， 正 则 表达 式 “^.+@.+\ .com$” 可 以 匹配 如 下 哪 一 个 字符 串 ? 


a) you@example.com b) 1943@example.net 
c) Q@example.com d) youremailaddress@example.org 


来 分 析 一 下 这 个 正则 表达 式 : 

口 “^+” 表 示 以 一 个 任 一 字符 开头 的 任意 长 度 的 字符 串 ， 

口 “@.+” 表 示 一 个 接 在 “@” 后 的 任意 长 度 的 字符 串 ， 

口 “\com$” 表 示 一 个 以 “.com” 结 尾 的 字符 串 。 

把 它们 三 个 连 在 一 起 ， 就 是 以 任 一 字符 开头 ， 包 含 “@”， 并 以 “.com” 结 尾 的 任意 
长 度 的 字符 串 。 说 白 了 它 匹配 的 就 是 以 “.com” 结 尾 的 邮箱 地 址 。 而 选项 b 和 选项 d 都 不 
是 以 “.com” 结 尾 的 ， 可 以 排除 了 。 而 选项 就 错 在 它 是 以 “@” 开 头 的 。 所 以 只 有 选项 
a 才 是 正确 的 。 

现在 可 以 写 一 段 脚本 来 验证 一 下 我 们 的 分 析 是 否 正确 。 在 这 段 脚 本 中 ， 会 用 到 ereg0 
函数 ， 它 带 有 如 下 两 个 参数 : 


preg match("pattern", $variable) 


其 中 pattem 指 的 是 正则 表达 式 ， 注 意 一 定 要 用 一 对 引号 把 正则 表达 式 包 右 起 来 。 
$variable 则 代表 指定 的 变量 。 用 于 检测 的 脚本 如 下 : 
【 例 7.2】 检测 字符 串 与 正则 表达 式 是 否 匹 配 。 


下 $arrayl = "Hello World!™"; 
区 $array2 = "MY Dear Kim"; 
3 $array3 = "9001"7 


ss 
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4 Sarray4 = "7851-32564"; 

5 S$array5 = " you@example.com"; 

6 

学 if (preg match("/^[A-Za-z].*/", $arrayl)) { 

8 echo "$arrayl matches the regular expression."; 

9 } else { 

10 echo "$arrayl does not matches the regular expression.™"; 
Lu 

12 

13 if (preg match ("/Dear (Kim | Jim)/", $array2)) { 

14 echo "$array2 matches the regular expression."; 

15 } else { 

16 echo "$array2 does not matches the regular expression."™; 
Ti 

18 

L900 (pregimatch /0 oA (NN L0H S/Sarray3d 

20 echo "$array3 matches the regular expression."; 

21 } else { 

22 echo "$array3 does not matches the regular expression."; 
3 

24 

25 if (preg match ("/^[0-9]{4}(\-[0-9]{5})$/", $array4)) { 

26 echo "$array4 matches the regular expression."; 

27 } else { 

28 echo "$array4 does not matches the regular expression."; 
29 于 

30 

31 FE (preg natch :ecTMICons/ SarrayS)) { 

32 echo "$array5 matches the regular expression."; 

33 } else { 

34 echo "$array5 does not matches the regular expression."; 
35° °° } 


这 段 脚 本 实在 太 长 了 ， 而 且 重 复 的 内 容 很 多 。 如 果 我 们 把 所 有 的 条 件 都 放 在 一 起 就 好 了 。 

其 实在 7.1.1 小 节 中 ， 我 们 就 提 过 ， 可 以 一 次 性 定义 多 个 条 件 ， 然 后 用 “和 ”或 “或 ” 
把 这 些 条 件 连接 起 来 。 用 脚本 应 该 怎么 写 呢 ? 

在 PHP 脚本 中 ， 我 们 可 以 使 用 and、or 或 xor 来 连接 两 个 条 件 。 其 中 : 

口 and 表示 两 个 条 件 均 为 真 (tue) 时， 复合 条 件 为 真 ; 

口 or 表示 两 个 条 件 中 任 一 条 件 为 真 或 两 个 条 件 均 为 真 (tue) 时 ， 复 合 条 件 为 真 ; 

口 xor 表示 两 个 条 件 中 任 一 条 件 为 真 (tue) 时， 复合 条 件 为 真 。 

使 用 这 些 连 字符 ， 我 们 可 以 把 例 7.2 所 示 脚 本 的 第 7 行 到 第 35 行 的 代码 改 成 如 下 的 样子 : 

J if (preg match("/^[A-Za-z].*/", $arrayl) && 

8 preg match ("/Dear (Kim | Jim)/", $array2) && 

9 preg match ("/^[0-9] {4} (\-[0-9] {5})$/", $array3) && 


10 preg match ("/^[0-9] {4} (\-[0-9] {5})$/", $array4) && 
11 preg match ("/^.+@.+\.com$/", $array5)) 


Ee 

3 echo "These arrays match their corresponding regular expressions."; 
14 } else { 

5 echo "Either certain or all arrays do not match their corresponding 
regular expressions -"7 

L600} 


在 这 短 短 十 行 的 代码 里 ， 我 们 使 用 and 把 这 五 个 preg_match0) 函 数 连接 了 起 来 。 也 就 
是 说 ， 只 有 当 这 五 个 ereg0 函 数 返 回 的 值 均 为 tue 时 ， 才 会 打印 如 下 这 名 话 : 


These arrays match their corresponding regular expressions. 
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否则 ， 系 统 会 输出 : 


Either certain or all arrays do not match their Corresponding regular 
expressions. 


7.1.3 ”简单 条 件 语句 if...else... 


通过 前 两 个 小 节 的 讲解 ， 大 家 对 这 个 单条 件 语句 肯定 已 经 不 陌生 了 。 在 本 小 节 里 ， 我 
们 将 系统 地 来 讲 讲 这 个 单条 件 语句 。 它 的 结构 如 下 。 


例 7.3】 简单 条 件 语句 寺 .. else... 的 结构 。 


if (conditionl) { 


} 
k 
} 


statement block A 
elseif (condition2) { 
statement block B 

else { 
statement block C 


如 例 7.3 所 示 ， 简 单条 件 语 句 站 由 三 个 部 分 构成 。 如 果 语 句 块 A、 语句 块 B 和 语句 块 
C 中 都 只 有 一 条 语句 的 话 ， 那 么 花 括号 也 是 可 以 去 掉 的 。 


( 


1) 站: 这 一 部 分 是 必 选 的 ， 用 于 指定 判断 条 件 。 


口 如 果 条 件 的 判断 结果 为 tue: 系统 执行 语句 块 A 中 语句 。 在 语句 块 A 中 的 语句 执 


行 完 毕 后 ， 系 统 将 直接 执行 语句 块 C 之 后 的 内 容 ， 语 句 块 B 和 语句 块 C 中 的 语句 
被 全 部 忽略 了 。 


口 如 果 条 件 的 判断 结果 为 false: 语句 块 A 中 的 语句 则 被 忽略 ， 系 统 将 转 而 执行 紧 随 


( 


其 后 的 语句 。 需 要 说 明 的 是 ， 紧 随 其 后 的 可 以 是 elseif 语句 、else 语句 或 其 他 的 
语句 。 


2) elseif 这 一 部 分 是 可 选 的， 用 于 指定 另 一 个 判断 条 件 。 


口 如 果 条 件 的 判断 结果 为 tue: 系统 执行 语句 块 B 中 的 语句 。 在 语句 块 B 中 的 语句 


执行 完毕 后 ， 系 统 将 直接 执行 语句 块 C 之 后 的 内 容 。 语 名 块 C 中 的 语句 则 被 忽 
略 了 。 


口 如 果 条 件 的 判断 结果 为 false: 语句 块 B 中 的 语句 则 被 忽略 。 系 统 将 转 而 执行 紧 随 


其 后 的 其 他 语句 。 需 要 说 明 的 是 ， 紧 随 其 后 的 可 以 是 另 一 个 elseif 语句 、else 语句 


或 其 他 语 语句 。 
3) else: 这 一 部 分 也 是 可 选 的 。 只 有 当 它 前 面 的 一 个 条 件 判断 为 false 时 ， 这 一 部 分 


中 包含 的 语句 才 会 被 执行 。 在 例 7.3 中 ， 只 有 当 elseif 中 的 条 件 判断 为 false 时 ， 语 句 块 C 
中 的 语句 才 会 被 执行 。 一 个 条 件 语 句 只 能 含有 一 个 else 部 分 。 如 果 在 一 个 条 件 语句 中 使 用 
了 else 部 分 ， 该 条 件 语句 必须 以 else 部 分 作 结 束 。 

现在 来 看 一 个 例子 。 


腿 设 你 是 一 位 老师 ， 需 要 将 100 名 学 生 的 百分制 成 绩 转 换 成 五 级 制 ， 并 将 转换 后 的 成 


绩 输 
【 
下 


要 


上 到 浏览 器 告知 学 生 。 我 们 应 该 怎么 做 呢 ? 
例 7.4】 简单 条 件 语句 示例 。 
<?php 


SactualScore = 87; 
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二 
2 
13 


14 
下 5 
16 
汪汪 
18 
20 
El 
E44 
3 


//8 


9 


if ($actualScore > 92){ //92 分 以 上 

SOrade = Ans 

Smessage = "Well Done!™; 
} elseif ($actualScore <= 92 and S$actualScore > 80) { 
0=924( 售 922 分 

$grade = "B"; 

Smessage = "Good! Keep on trying!'"; 
} elseif ($actualScore <= 80 and $actualgrade > 70) { 
0~80 ( 含 80) 分 

$grade = "C"; 

Smessage = "Okay! Not bad!"; 
} elseif ($actualScore <= 70 and S$actualScore > 60) { 


//60~70 ( 含 70) 分 


Es 


$grade = "D"™; 


$message = "Uh oh! You've just won the game!"; 

TS //60 及 60 分 以 下 
Sgrade = "E"7 
Smessage = "Try harder next time!"; 


} 


echo "$message<br>"; 
echo "You grade is S$grade."™; 


在 例 7.4 这 段 脚本 中 ， 定 义 了 一 个 变量 $actualScore 用 于 存放 学 生 的 实际 分 数 ， 然 后 将 
实际 分 数 与 转换 标准 做 比较 : 


口 


口 


口 


略 过 


际 分 数 高 于 92 分 时 ， 学 生 的 等 级 为 A， 评 语 为 “Well Done!”。 这 时 系统 会 
第 7 到 第 20 行 的 内 容 ， 直 接 执行 第 21 行 : 输出 评语 和 等 级 。 


当 实 际 分 数 低 于 92 分 但 高 于 80 分 时 ， 学 生 等 级 为 B， 评 语 为 “Good! Keep on 
trying!”。 这 时 系统 会 略 过 第 5 行 、 第 6 行 以 及 第 10 到 第 20 行 ， 直 接 执 行 第 21 
行 : 输出 评语 和 等 级 。 

当 实 际 分 类 低 于 80 分 但 高 于 70 分 时 ， 学 生 等 级 为 C， 评语 为 “Okay! Not bad!”。 
这 时 系统 会 略 过 第 5 到 第 9 行 以 及 第 13 到 第 20 行 ， 直 接 执行 第 21 行 : 输出 评语 
和 等 级 。 

当 实 际 分 类 低 于 70 分 但 高 于 60 分 时 ， 学 生 等 级 为 D， 评 语 为 “Uh oh! You've just 


Won 


the game!”。 这 时 系统 会 略 过 第 5 到 第 12 行 以 及 第 16 到 第 20 行 ， 直 接 执行 


第 21 行 : 输出 评语 和 等 级 。 


xx 二 


司 关 
时 系 


际 分 数 低 于 60 分 时 ， 学 生 的 等 级 为 F， 评 语 为 “Try harder next time!”。 这 
统 会 略 过 第 5 到 15 行 ， 直 接 执 行 第 21 行 : 输出 主语 和 等 级 。 


在 例 7.4 中 ， 假 定 某 位 学 生 的 实际 成 绩 为 87 分 。 那 么 执行 的 结果 应 该 如 图 7-2 所 示 。 


. 134 


ipV/wwwgre [ee_ mm me 


人 249 HRT TT CE 


突 Favorites 。 车 http://www.greatwalltea/chapte/07/EXT-04.php 和食-~ 国 -局 吉 > Pagev safeyv Toosv 属 - 


‖ 6o0 
You 


dl Keep on trying! 
grade is B. 


图 7-2 例 7.4 的 执行 结果 
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读者 可 以 通过 修改 变量 SactualScore 的 值 来 改变 脚本 执行 的 结果 。 

有 的 时 候 ， 情 况 会 变 得 有 些 复 杂 : 当 一 个 条 件 判断 为 真 时 ， 还 需要 再 做 出 一 个 判断 ， 
才能 得 到 想 要 的 结果 。 比 如 例 7.5。 

【 例 7.5】 条 件 语 句 的 嵌 套 的 使 用 。 


<?php 
IE (SCcustProv ==,. “HB™) { 
if ($isEmailAdd == ""){ 
$contactMethod = "letter"; 
} else { 
$contactMethod = "Email"; 
} 
} else { 
$contactMethod = "Not for this time"; 
} 
x 


这 段 脚本 先 对 顾客 所 在 的 省 份 进行 了 判断 : 如 果 顾 客 不 是 来 自 湖北 省 ， 则 忽略 之 〈 直 
接 为 变量 $contactMethod 赋值 “Not for this time”) 。 若 顾客 来 自 湖北 省 ， 则 继续 判断 其 是 
否 登记 了 电邮 地 址 。 若 其 电邮 地 址 为 空 ， 则 向 其 发 送 函 件 (为 变量 $contactMethod 赋值 
“Letter”); 若 登 记 了 电邮 地 址 , 则 向 其 发 送 电子 邮件 (为 变量 gcontactMethod 赋值 <Email”)。 
在 这 段 脚本 中 ， 我 们 用 了 两 个 条 件 语句 ， 其 中 一 个 嵌 套 在 了 另 一 个 中 间 。 这 就 叫 条 件 语 名 
的 嵌 套 。 


7.1.4 复杂 条 件 语 句 switch 


在 简单 条 件 语句 中 ， 我 们 遇 到 的 始终 都 是 二 选 一 的 情况 ， 非 黑 即 白 。 然 而 在 现实 生活 
中 , 我 们 经 常会 面临 多 选 一 的 抉择 .用 PHP 脚本 来 描述 这 种 多 选 一 的 情况 , 就 得 用 到 switch 
它 的 结构 如 下 : 
Switch ($varName) 
{ 
case $valuel : 
Statement block A; 
break; 
case $value2 : 
statement block B; 
break; 
case $value3 : 
statement block C; 
break; 


default: 
statement block by default; 
break; 


} 

在 这 一 段 脚本 中 ，PHP 处 理 引擎 会 测试 变量 SvarName 的 值 ， 然 后 根据 SvarName 的 值 
跳 到 相应 的 case 部 分 执行 相应 代码 块 中 的 代码 直到 遇 到 break 语句 或 switch 语句 末端 。 若 
被 测试 变量 的 值 没有 匹配 任何 case 部 分 ，PHP 处 理 引 擎 会 直接 执行 默认 部 分 代码 。 一 个 
switch 语句 中 case 部 分 的 数量 由 你 来 定 ， 而 默认 部 分 的 则 是 可 选 的 。 阁 在 switch 语句 中 添 
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加 了 默认 部 分 ， 则 最 好 将 其 放 在 所 有 case 部 分 的 后 面 。 
下 面 这 段 脚 本 针对 不 同 的 省 份 ， 设 置 了 不 同 的 简称 。 一 起 来 看 看 它 是 怎么 做 到 的 。 
【 例 7.6】 为 省 份 设置 简称 。 

Switch ($province) 
{ 
case "Beijing": 
SabbrValue = "BJ"; 
break; 
case "Hebei": 
$abbrValue = "HEB"; 
break; 
case "Henan": 
$abbrValue = "HEN"; 
break; 
case "Hubei": 
SabbrValue = "HB"; 
break; 
default: 
$abbrValue = "Undefined"; 
break; 


} 
在 例 7.6 中 , 使 用 了 switch 语句 来 测试 变量 $province 的 值 。 并 提供 了 五 种 可 能 的 取 值 : 
若 变量 $province 的 取 值 为 “Beijing”， 则 变量 $abbrValue 的 取 值 为 “BJ”。 
若 变 量 $province 的 取 值 为 “Hebei”， 则 变量 $abbrValue 的 取 值 为 “HEB”。 
若 变量 $province 的 取 值 为 “Henan”， 则 变量 $abbrValue 的 取 值 为 “HEN”。 
若 变量 $province 的 取 值 为 “Hubei”， 则 变量 $abbrValue 的 取 值 为 “HB”。 
车 变量 $province 的 值 未 匹配 上 述 任 一 取 值 ， 则 变量 $abbrValue 的 取 值 为 
“Undefined”。 

在 使 用 switch 语句 时 ， 需 要 特别 注意 ， 在 结束 每 个 case 部 分 时 ， 一 定 要 使 用 break 语 
句 ， 告 诉 PHP 处 理 引 擎 switch 语句 中 余下 的 内 容 不 用 处 理 了 ， 和 否则 PHP 处 理 引 擎 会 顺序 
执行 switch 语句 中 余下 的 部 分 。 我 们 还 是 以 例 7.6 为 例 ， 若 把 其 中 的 break 语句 全 部 去 掉 ， 
无 论 变量 $province 的 取 值 是 什么 ， 变 量 $abbrValue 的 取 值 均 为 “Undefined”。 


OOODODD 


7.1.5 实战 练习 : 用 户 信息 验证 


在 之 前 几 个 章节 的 实战 练习 中 ， 我 们 都 反复 用 到 条 件 来 控制 脚本 的 运行 。 其 实 ， 对 于 
一 个 稍微 有 些 用 处 的 PHP 脚本 来 说 ， 使 用 几 个 条 件 判断 和 循环 往复 来 控制 脚本 的 运行 ， 是 
一 件 稀 松平 常 的 事情 。 在 这 一 节 里 ， 我 们 将 巩固 本 章 里 学 到 的 内 容 来 完成 一 项 任务 ， 那 就 
是 验证 用 户 在 表单 中 输入 的 各 项 信息 是 否 符合 要 求 。 

看 到 这 里 ， 估 计 有 的 同学 已 经 明白 是 怎么 一 回 事 儿 了 。 是 的 ， 我 们 要 使 用 正则 表达 式 
对 用 户 输入 的 各 项 信息 进行 检查 ， 以 防止 不 符合 要 求 的 数据 影响 系统 的 稳定 性 。 

在 用 户 填写 完成 图 7-3 左 侧 的 表单 ， 并 单 击 “ 提 交 ” 按 钮 后 ， 在 页 面 右 侧 会 出 现 相 应 
的 信息 。 如 果 用 户 有 未 填写 的 信息 ， 或 者 填写 的 信息 不 符合 格式 要 求 ， 均 会 出 现 相应 的 提 
示 。 只 有 当 用 户 填写 的 信息 完全 符合 格式 要 求 时 ， 才 会 出 现 如 图 7-3 所 示 的 内 容 。 
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站 名 张 先 生 ， 您 好 
您 出 生 于 1985 年 06 月 27 日 * 
我 们 可 以 通过 向 cver@example com 发 送 电子 邮件 或 拨打 13910482222 联 系 到 您 。 


性 别 夯 男 目 女 
出 生年 月 
电子 邮箱 


@ Intermet | Protected Mode Off 给 > 护 100% ~ 


图 7-3 用 户 信息 验证 


我 们 来 看 看 具体 的 实现 过 程 。 
(1) HTML 部 分 。 


<!DOCTYPE html> 
<html> 
<head> 
<meta charset='UTF-8'> 
<title> 用 户 信息 验证 </title> 
<style> 
body { 
font-size:small; 


} 


form div { 
line-height: 35px; 
} 


.form-container { 
width:30%; 
float:left; 

} 


.info-cotainer { 
width:80%; 
float:left; 

时 


</style> 
</head> 
<body> 

<h3> 用 户 信息 验证 </h3> 

< > 

<div class="form-container"> 

<form action="?check" method="post"> 
<div> 
姓名 <input type="text" size="6" name="user"> 


ss 
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</div> 
<div> 
性 别 
<input type="radio" name="gender" value="male" checked> 男 
<input type="radio" name="gender" value="female"> 女 
</div> 
<div> 
出 生年 月 
<input type="text" size="12"” name="birthday"> 
</div> 
<div> 
电子 邮箱 
<input type="text" name="email"> 
</div> 


<input type="text" size="10" name="phone"> 
</div> 


<button type="submit"> 提 交 </button> 


</form> 
</div> 
<div class="info-container"” id="info"></div> // 信 息 会 输出 到 这 里 
</body> 
</html> 


在 这 个 页 面 里 ， 我 们 使 用 CSS+DIV 布局 把 页 面 分 成 了 三 个 部 分 ， 一 个 用 于 存放 “用 
户 信息 验证 ”标题 、 一 个 用 于 位 于 标题 下 方 左 侧 的 表单 、 另 一 个 则 用 于 存放 标题 下 方 右 侧 
的 信息 展示 区 域 。 

我 们 需要 实现 的 功能 是 ， 验 证 用 户 在 标题 上 方 左 侧 的 表单 中 输入 的 数据 是 否 符合 要 
求 。 若 符合 要 求 ， 则 在 标题 下 方 右 侧 的 信息 展示 区 域 中 输出 相应 的 信息 ， 若 不 符合 要 求 ， 
则 在 标题 下 方 右 侧 的 信息 展示 区 域 中 输出 错误 提示 信息 ， 提 醒 用 户 修改 表单 中 的 数据 。 

(2) PHP 部 分 。 

<?php 


if(isset($ REQUEST['check'])){ // 判 断 用 户 是 否 提交 了 表单 
$name = $ REQUEST['user']; 
S$gender = $ REQUEST['gender']; 
$birthday = $ REQUEST['birthday']; 


Semail = $ REQUEST['email']; 


Sphone = $_REQUEST['phone'];  ”// 获 取 用 户 输入 的 信息 
Smsg = 7 // 定 义 输出 消息 的 初始 值 
Fioname = 

S$gender == "'" || 

$birthday == "" || 

Semail == "'， || 

$phone == "'"'){ // 判 断 是 否 有 未 输入 的 表单 项 


Smsg = '<1i> 您 至 少 有 一 项 没有 填写 。</1i>'; 
Helseif (Ipregumateh(s /SLO T1437 NA 1100 -91 WTIO 31L0 91937 
Sbirthqay) ){ 
$msg = "<1i> 您 输入 的 出 生日 期 格式 不 正确 。 可 接受 的 格式 有 : 
<ol><1i>2013-07-06</1i><1i>2013/07/06</1i><1i>2013.07.06</1i> 
/ol></ i 
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/* 上 面 的 正则 表达 式 是 判断 用 户 输入 的 是 否 为 可 接受 的 三 种 格式 之 一 。 */ 

} elseif (!preg match('/^.+@.+\.com$/',$email)){ 
Smsg ="'<1i> 您 输入 的 邮件 地 址 格式 不 正确 。 可 接受 的 格式 为 : you@example. com。 
i 

/* 上 面 的 正则 表达 式 是 判断 用 户 输入 的 邮件 地 址 格式 是 否 正确 。*/ 

} elseif (!Preg match('/^1[3458] [0-9] {9}$/',$phone)){ 
Smsg ='<1i> 您 输入 的 手机 号 码 格式 不 正确 。 现 只 接受 中 国 大 陆地 区 的 手机 号 码 。 


/I 
/* 上 面 的 正则 表达 式 是 判断 用 户 输入 的 手机 号 码 是 否 为 13、14、15、18 开头 的 中 国 大 陆 
地 区 的 手机 号 码 */ 
} else { 
$title = mb substr ($name,0,1,'utf-8°'); 
if($gender == 'male') { 
$title .= "先生 "> 
} else { 
Stitle = 
| // 获取 用 户 的 称谓 


Smsg = '<p>' .$title.'， 您 好 ! </p>"'; 
$msg -= '<p> 您 出 生 于 ' .substr ($birthday,0,4).' 年 ' .substr 
($birthday, 5,2).' 月 '. 

substr ($birthday, 8,2) .' 昌 H. </p>'; 
$msg .= '<p> 我 们 可 以 通过 向 ' .$email.' 发 送 电 子 邮 件 或 拨打 ' .$phone.' 联 
系 到 您 。</p>'; 


echo '<script language="javascript">document .getElementById 
("info") .innerHTML="<ul>"'. 


$msg.'</ul>"</script>"; // 输出 用 户 的 信息 到 页 面 中 指定 的 位 置 
} 


?> 


在 上 面 这 段 PHP 脚本 中 ， 依 然 通过 “?check” 这 样 一 个 URL 后 缕 来 判断 用 户 是 否 提 
交 了 表单 (关于 如 何在 页 面 问 传递 数据 ， 请 参考 本 书 9.1 节 的 内 容 ) 。 


49 用户 提交 了 表单 ， 我 们 将 使 用 $_REQUEST 数组 把 用 户 输 入 的 内 容 存放 在 $name、 


$gender、$birthday、$email 和 $phone 五 个 变量 中 。 然 后 再 定义 一 个 空 字符 串 变量 gmsg 用 
于 存放 在 对 以 上 五 个 变量 验证 后 产生 的 信息 。 车 用 户 输 入 的 信息 符合 要 求 ， 则 将 用 户 输 入 


重新 整合 后 存 入 变量 Smsg 中 ; 若 用 户 输入 的 信息 不 符合 要 求 , 则 将 提示 用 户 修改 数 


据 的 信息 存 入 变量 $msg 中 。 
按照 上 述 要 求 ,我们 在 脚本 中 使 用 了 “if...elseif...else...” 结 构 和 正则 表达 式 实 现 了 对 
用 户 输入 数据 的 验证 工作 。 


在 这 里 ， 


我 们 来 详细 了 解 一 下 脚本 中 使 用 的 正则 表达 式 : 


口 “[0-9]{4}[V.][0-1][0-9][-V.][0-3][0-9]$ 


这 一 条 


E 则 表达 式 用 通俗 的 话 来 说 , 就 是 一 个 用 来 判断 年 月 日 格式 的 正则 表达 式 。 其 中 ， 


年 份 必须 为 四 位 ， 月 份 和 日 期 必须 为 两 位 。 年 、 月 、 日 之 间 由 “-”、“/” 或 “.” 分 隔 。 


口 ^+@.+t\.com$ 
这 一 条 正则 表达 式 则 是 用 来 判断 电子 邮件 地 址 的 ， 具 体 可 以 参考 7.1.1 小 节 中 的 内 容 。 


口 ^1[3 


458][0-9]{9}$ 


这 一 条 正则 表达 式 则 规定 了 一 串 以 “1” 开头 的 11 位 数字 串 , 并 且 第 二 位 必须 为 3、4、 


i 
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5 或 8。 

当 用 户 输 入 的 任何 一 条 数据 为 室 ， 或 者 无 法 匹配 上 述 的 正则 表达 式 ， 则 会 导致 系统 在 
标题 下 方 右 侧 输入 对 应 的 提示 信息 。 而 只 有 当 用 户 输入 了 所 有 要 求 的 信息 ， 且 生日 、 电 子 
邮箱 地 址 和 手机 号 码 均 匹 配对 应 的 正则 表达 式 ， 系 统 才 会 在 标题 下 方 右 侧 的 页 面 中 输出 类 
似 图 7-3 所 示 的 信息 。 


7.2 重复 性 运算 一 一 循环 


在 这 一 章 的 开始 ， 我 们 就 说 过 : 在 实际 生活 中 ， 除 了 需要 不 断 地 做 出 选择 之 外 ， 我 们 
还 得 不 断 重复 着 某 些 活动 。 无 论 你 能 否 接受 ，C'est la vie。 

为 了 模拟 这 个 重复 的 过 程 ， 也 为 了 让 脚本 变 得 更 加 轻巧 ，PHP 给 出 了 解决 重复 性 运算 
的 方案 ， 那 就 是 循环 。 在 PHP 脚本 中 ， 可 以 使 用 三 种 类 型 的 循环 ， 如 下 所 示 。 

口 for 循环 : 在 一 个 for 循环 中 ， 需 要 设置 一 个 计 步 器 和 一 段 需要 重复 执行 的 脚本 。 
当 计数 器 的 值 达到 指定 上 限时 ， 跳 出 循环 。 

口 while 循环 : 在 一 个 while 循环 中 ， 需 要 设置 一 个 条 件 供 PHP 处 理 引 擎 检测 。 若 条 
件 判断 为 tue，PHP 处 理 引擎 重复 执行 某 一 段 脚本 直到 条 件 判断 为 false 时 跳出 
循环 。 

口 do...while 循环 : 在 一 个 do...while 循环 中 ， 将 先 执行 一 段 脚本 ， 然 后 设置 一 个 条 
件 供 PHP 处 理 引擎 检测 ， 若 条 件 判断 为 tue，PHP 处 理 引 擎 重复 执行 之 前 执行 的 
那 一 段 脚本 直到 条 件 判断 为 false 时 跳出 循环 。 

在 本 节 里 ， 我 们 就 来 看 看 在 PHP 脚本 中 ， 应 该 如 何 使 用 这 些 循 环 。 


7.2.1 for 循环 


上 面 说 过 ， 最 基本 的 for 循环 需要 定义 一 个 循环 变量 〈 计 步 器 ) ， 并 为 这 个 变量 设置 
起 始 值 、 终 结 条 件 和 步 长 。 基 本 结构 如 下 : 

for (start value; end condition; increment) 

{ 
statement block 

} 

其 中 ， 三 个 变量 之 间 是 用 两 个 分 号 (;) 隔 开 的 ， 第 三 个 参数 后 面 没 有 分 号 。 

口 start value 定义 了 循环 变量 的 起 始 值 。 如 果 在 这 参数 中 写 上 “$i= 1”， 那么 循环 变 
量 就 是 Si， 这 个 变量 的 起 始 值 为 1。 通 常情 况 下， 我们 的 循环 变量 都 是 用 0 或 1 做 
为 起 始 值 的 ， 但 是 也 可 以 使 用 其 他 数字 或 变量 来 定义 循环 变量 的 起 始 值 。 

口 end condition 定义 了 循环 变量 的 终结 值 。 如 果 在 这 个 参数 中 写 上 “$i<=10”， 那 么 
循环 变量 的 终点 为 10。 只 要 终结 条 件 为 tue， 循 环 就 一 直 持 续 下 去 。 只 有 当 终 结 
条 件 为 false 时 ， 循 环 才 会 终结 。 循 环 变量 的 终结 值 可 以 是 数 ， 也 可 以 是 一 个 
变量 。 

口 increment 定义 了 循环 变量 的 步 长 ， 也 就 是 如 何 从 起 始 值 一 步 一 步 走 到 终结 值 的 。 
如 果 在 这 个 参数 中 写 上 “i++” 或 者 “二 +1”， 那 么 这 个 循环 的 步 长 为 1， 也 就 是 
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说 每 循环 一 次 ， 循 环 变量 加 1。 通 常情 况 下 ， 我 们 用 设置 循环 的 步 长 为 1， 不 过 也 
可 以 使 用 其 他 的 步 长 。 

口 statement block 定义 了 需要 重复 执行 的 语句 。 也 就 是 说 循环 变量 的 值 每 改变 一 次 ， 
该 语句 块 中 的 语句 都 会 被 执行 一 次 。 如 果 我 们 在 这 里 写 上 “echo "Hello World!";”， 
那么 只 要 终结 条 件 为 tue，PHP 处 理 引擎 就 会 不 断 的 输出 “Hello World!” 这 人 句 话 。 

我 们 把 上 面 四 段 文字 的 解析 写 在 一 起 就 是 下 面 这 个 样子 。 

【 例 7.7】 一 个 简单 的 循环 。 

Eo i 

: eehonm mre 

echo "Hello World!"; 


echo "<br>"; 
) 


这 段 脚 本 会 以 一 行 一 句 的 形式 输出 10 句 “Hello World!”， 如 图 7-4 所 示 。 


[cle = oj] 


hapy/wwwarea 


hatpvrwww .greatwalltea/chapteo7yED-07.php - 图 四 四 Bing Pr 


会 Favorites | 轿 htpy/wwwgreatwalltea/chapterI7/EI-07.php 租 ~ 国 -加 出 Pagev sfeyv Toosv 加 


1. Hello World! 
2. Hello World! 
3. Hello World! 
4. Hello World! 
5. Hello World! 
6. Hello World! 
7. Hello World! 
8. Hello World! 
9. Hello World! 
10. Hello World! 


图 7-4 循环 输出 Hello World! 


for 循环 还 支持 嵌 套 ， 也 就 是 说 我 们 可 以 在 一 个 循环 中 进行 另 一 个 循环 。 比 如 ， 如 果 要 
出 九 九 乘法 表 的 话 ， 我 们 可 以 这 么 写 。 
【 例 7.8】 九 九 乘法 表 。 


于 <?php 
芭 echo "<table border='0' cellspacing='5' cellpadding="'5'>"; 


个 for($x = 1; $x < 10; $x++){ 

4 echo "<tr>"; 

六 for($y=1; $y < 10; S$y++){ 

6 if($x >= $y){ 

yi! echo "<td>$y x $x = ".$x*$y."</td>"; 
8 


} else { 
全 echo "<td></td>"; 
10 
于 入 J 
1 人 2 echo "</tr>"; 
hi } 
14 echo "</table>"; 
1 
输出 的 结果 如 图 7-5 所 示 。 


在 这 段 脚本 中 ， 使 用 了 for 循环 的 嵌 套 。 在 第 一 个 for 循环 中 ,我 们 定义 了 变量 8x， 起 


“141: 


第 3 篇 ”PHP 编程 基础 


始 值 为 1， 步 长 为 1， 终结 值 为 9。 在 第 二 个 for 循环 中 ， 我 们 定义 了 变量 Sy， 起 始 值 、 步 
长 和 终结 值 与 变量 $x 相同 。 


ee . I EE 


会 Favorites 。 车 htpy/www.greatwalltea/chapter7/EI-08.php 租 ~ 国 "可 于 Pagev safey” Toosv 各- 


2x2=4 

2x3=6 3x3=9 

2x4=8 3x4=12 4z4=16 

2x5=10 3x5=15 4x5=20 5x5=25 

2x6=12 3x6=18 4x6=24 5x6=30 6x6=36 

2xz7=14 3x7=21 4x7=28 5x7=35 6xz7=42 7xz7=49 

2x8=16 3x8=24 4x8=32 5x8=40 6x8=48 7x8=56 8x8=64 
2x9=18 3x9=27 4x9=36 5x9=45 6x9=54 7x9=63 8x9=72 9x9=8l 


@ Intemet | Protected Mede Off 广 - 所 100% ~ 


图 7-5 九 九 乘法 表 


在 这 段 脚 本 中 ， 外 层 循环 只 会 被 执行 9 次 ， 而 内 层 循 环 会 在 外 层 循环 每 执行 一 次 时 就 
执行 9 次 ， 共 计 81 次 。 我 们 一 起 来 看 看 这 段 程序 是 按照 什么 顺序 执行 的 : 

PHP 处 理 引 擎 首先 执行 外 层 的 for 循环 ， 获 取 变 量 $x 的 值 。 继 续 往 下 执行 。 当 碰 到 内 
层 的 for 循环 时 ， 获 取 变 量 $y 的 值 。 在 内 层 的 for 循环 中 ， 我 们 还 使 用 了 一 个 辽 条 件 句 ， 
用 来 比较 变量 $x 和 变量 $y 之 间 的 大 小 。 当 变量 $x 的 值 大 于 等 于 变量 $y 的 值 时 ， 打 印 变量 
$x 和 S$y 的 乘积 。 当 变量 $x 的 值 小 于 变量 $y 的 值 时 ， 打 印 空 单元 格 。 

在 这 例 7.8 中 : 

口 当 变 量 $x 和 变量 $y 均 为 1 时 , 打印 “1x1=1”; 

口 当 变 量 $x 为 1， 变量 $y 为 2 时 ， 打 印 空 单元 格 ; 

口 当 变 量 $x 为 2， 变量 $y 为 1 时 ， 打 印 “1x2=2”; 

口 当 变 量 $x 为 2， 变量 $y 也 为 2 时 ,打印 “2x2=4”; 


以 此 类 推 ， 得 到 如 图 7-5 所 示 的 结果 。 

其 实 上 面 的 例子 对 于 PHP 来 说 只 是 小 儿科 。 更 为 复杂 的 循环 应 该 是 下 面 这 个 样子 的 : 
for(initial statement; 

loop condition; 


end-loop statement) 


{ 


statement blocks 


| 

其 中 ， 

口 initial statement 定义 了 循环 起 始 状 态 。 这 一 部 分 的 语句 只 在 循环 开始 时 执行 一 次 。 
口 loop condition 定义 了 循环 继续 的 条 件 。 若 条 件 判 断 为 false， 则 循环 终止。 

口 end-1oop condition 定义 了 完成 每 一 次 循环 后 需要 执行 的 语句 。 
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上 面 三 项 之 间 用 两 个 分 号 (; ) 隔 开 ， 第 三 项 后 面 没 有 分 号 。 
再 来 看 一 个 例子 。 
【 例 7.9】 复杂 循环 与 简单 循环 。 


1 <table border="0" cellspacing="5" cellpadding="5"> 


全 Er 

3 <td>Even Numbers</td> 
4 <?php 

5 // 打 印 10 以 内 的 偶数 

6 t= 0; 

人 ENSO $0 = 
8 St <= 8; 

9 $i++, $j++ 

10 A 

i $t = $i + $j; 

2 echo "<td>$t</tqd>"; 
i 民 1 } 

14 > 

JS </tr> 

16 FE 

<td>Even Numbers</td> 
18 <?php 

19 // 打 印 10 以 内 的 偶数 
20 Eor(Si = 0 $1 <= L102 S14=2) 4 
pa if ($i > 0) 

22 echo "<td>$i</td>"; 
23 } 

24 2> 

4 由 乓 中 


26 </table> 


在 例 7.9 这 段 脚本 中 ， 使 用 了 两 个 独立 的 for 循环 用 来 打印 10 以 内 的 奇偶 数 。 第 一 段 
for 循环 从 第 7 行 始 到 第 13 行 止 ， 是 一 个 复杂 循环 。 第 二 段 for 循环 从 20 行 始 到 第 23 行 
止 ， 是 一 个 简单 循环 。 分 别 来 看 看 这 两 段 脚本 : 

在 第 一 段 的 脚本 中 ， 我 们 在 循环 初始 定义 了 两 个 变量 ， 它 们 分 别 是 变量 $i 和 变量 $j。 
这 两 个 变量 的 值 均 为 1。 然 后 将 变量 $t 小 于 等 于 8 作为 循环 继续 的 条 件 ， 变 量 $t 的 原始 值 
为 0。 接 着 我 们 让 变量 $i 和 变量 $j 的 值 在 每 次 循环 后 分 别 加 1。 

在 每 次 循环 中 ， 都 将 变量 $i 和 变量 $j 的 和 赋值 给 变量 8t， 然 后 打印 变量 $t 的 值 到 指定 
单元 格 。 大 家 在 这 里 可 能 会 有 一 个 疑问 : 为 什么 这 一 段 脚本 最 后 会 输出 10 这 个 偶数 ? 循环 
难道 不 应 该 在 变量 $t 大 于 8 的 时 候 就 终止 了 吗 ? 

带 着 这 个 疑问 ， 我 们 再 来 看 看 这 一 段 脚本 。 假 设 此 时 变量 $i 和 变量 $j 均 为 4， 与 此 同 
时 变量 $t 的 值 应 该 为 6 而 不 是 8。 因 为 在 上 一 轮 的 循环 结束 前 ， 也 就 是 当 变 量 $i 和 变量 $j 
均 为 3 时 ，PHP 处 理 引擎 将 变量 $i 和 变量 $j 的 和 赋值 给 了 变量 gt。 因 此 当 变 量 $ 和 变量 $j 
为 4 时， 变量 $t 的 值 为 6。 在 这 一 轮 循环 结束 后 ， 变 量 $t 的 值 变 成 了 8， 而 变量 $i 和 变量 
$j 的 值 则 变 成 了 5。 由 于 此 时 变量 St 的 值 仍然 是 小 于 或 等 于 8 的 ， 于 是 循环 继续 ， 系 统 输 
出 变量 $i 和 变量 $j 的 和 ， 也 就 是 10， 然 后 变量 Si 和 变量 $j 各 自 加 1， 回 到 循环 起 点 。 这 时 
变量 $t 的 值 已 经 为 10， 循 环 继续 的 条 件 不 成 立 ， 循 环 结束 。 

在 第 二 段 脚 本 中 ， 我 们 在 循环 初始 定义 了 变量 $i， 并 为 其 赋值 0。 然 后 将 变量 $i 小 于 
等 于 10 作为 循环 继续 的 条 件 。 最 后 , 我 们 让 变量 $i 在 每 次 循环 结束 后 加 2。 在 每 次 循环 中 ， 
只 要 变量 $i 的 值 大 于 0 都 会 被 输出 。 于 是 就 看 到 了 一 串 小 于 10 的 偶数 。 
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例 7.9 所 示 中 的 两 段 脚 本 殊途同归 ， 分 别 展现 了 复杂 循环 和 简单 循环 的 不 同 魅 力 。 输 
出 结果 如 图 7-6 所 示 。 


htpy/wwwgres 


人 人 


请 Favorites | 赔 http//www.greatwalltea/chapter07/EI-09.php 从 ~ "加 出 Pagev Ssfetyv Toosv 加- 


Even Numbers 2 4 6 8 10 
Even Numbers 2 4 6 8 10 


图 7-6 复杂 循环 与 简单 循环 


7.2.2 while 循环 


在 7.2 节 的 开始 ， 我 们 说 过 : “在 一 个 while 循环 中 ， 需 要 设置 一 个 条 件 供 PHP 处 理 
引擎 检测 。 若 条 件 判断 为 tue，PHP 处 理 引 擎 重复 执行 某 一 段 脚本 直到 条 件 判断 为 false 时 
跳出 循环 。” 具 体 来 说 一 个 while 循环 有 两 个 部 分 : 

口 一 个 条 件 供 PHP 判断 ， 

口 一 段 需 要 循环 执行 的 脚本 。 

当 条 件 判 断 为 tue 时 ， 循 环 继续 ， 当 条 件 判断 为 false 时 ， 循 环 结束 。 

如 此 看 来 ，while 循环 的 结构 应 该 是 下 面 这 个 样子 : 


while (condition) 
| 


statement block 


有 


下 面 ， 用 while 循环 模拟 一 下 我 们 在 一 堆 铜 钮 扣 中 找 一 颗 金 钮 扣 的 过 程 吧 。 
【 例 7.10】 寻找 金 钮 扣 。 


和 <?php 

2 // 准 备 一 些 扣子 

3 Sbuttons = array("copperButton1l"， 

4 "copperButton2", 

上 "copperButton3", 

6 "copperButton4", 

7 "copperButton5", 

8 "goldButton", 

3 "copperButton6"); 

10 

11 // 准 备 一 些 在 找 不 着 金 钮 扣 时 需要 说 的 话 

12 $complaints = array ("Ooops! Not the gold one!<br>...<br>", 

3 "Uh oh! What a mess! where is the gold button? 
A oo Re) 

14 "Sigh! Still not the gold button! Where did I put 
站 

这 本 "Ahhha! This is it!!!! Uh, it seems a little bit 
different. Sigh!<br>...<br>"); 

16 
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1 // 准 备 一 些 在 寻找 金 钮 扣 过 程 中 需要 的 变量 


18 StestVar = "no™; 

和 $counter = 0; 

20 

21 // 开 始 找 金 钮 扣 

22 while (StestVar != "yes"){ 

23 $complaintRandom = mt rand(0, 3); 

24 IE ($buttons[$counter] == "goldButton"){ 
Ps StestVar = "yes"; 

26 echo "Hooray! The gold button is now mine!<br>"; 
把 新 } else { 

28 echo $complaints[$complaintRandom]; 
之 9 1 

30 $countertt+; 

Eh } 

3202> 


在 例 7.10 这 段 脚本 中 ， 定 义 了 一 堆 钮 扣 〈 数 组 Sbuttons) ， 其 中 只 有 一 颗 是 金 钮 扣 
(goldbutton) 。 然 后 又 定义 了 一 系列 抱怨 没有 找到 钮 扣 时 可 以 说 的 话 〈 数 组 $Scomplaints) 。 
紧 接着 定义 了 用 于 测试 当前 对 象 是 否 为 金 钮 扣 的 变量 StestVar 和 计 步 用 的 变量 $counter。 如 
果 当 前 找到 的 钮 扣 不 是 金 钮 扣 ， 那 么 变量 $testVar 的 值 不 变 。 如 果 当 前 找到 的 钮 扣 是 金 钮 
扣 ， 那 么 变量 $testVar 的 值 变 成 “yes”。 变 量 $counter 的 值 初始 为 0， 每 次 循环 结束 后 加 1。 

从 第 21 行 开始 ， 我 们 使 用 了 一 个 while 循环 ， 用 于 模拟 我 们 找 钮 扣 的 过 程 。 这 个 循环 
得 以 继续 的 条 件 是 变量 $testVar 的 值 不 为 “yes”。 由 于 循环 开始 时 变量 $testVar 的 值 不 为 
“yes”，PHP 开始 执行 while 循环 内 部 的 语句 。 

在 while 循环 内 部 使 用 了 一 个 直 语 句 用 来 判断 我 们 找到 的 是 不 是 金 钮 扣 : 如 果 是 ， 则 
将 变量 $testVar 的 值 改 为 “yes” 并 打印 “Hooray! The gold button is now mine!l<br>”; 如 果 
不 是 ， 则 随机 输出 一 句 抱怨 的 话 。 

若 在 某 次 循环 的 过 程 中 ， 变 量 $testVar 的 值 变 成 了 “yes”， 那 么 PHP 在 下 一 次 循环 开 
始 前 就 会 发 现 循环 继续 的 条 件 不 成 立 ， 循 环 结束 。 


CT EX | 
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帘 Favortes 臣 http//www.greatwalltea/chapter07/Ex7-10.php 全 7 加 吐 v Pagev Safetyv Toosv ©- | 


Qoops! Not the gold one! 

Sigh! Still not the gold button! Where did I put it? 

Sigh! Still not the gold button! Where did I put it? 

Ahhha! This is it!l!l!! Uh, it seens a little bit different, Sigh! 
th oh! What a mess! Where is the gold button? 


Hooray! The gold button is now mine! 


图 7-7 寻找 金 钮 扣 
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图 7-7 展示 了 该 脚本 执行 的 结果 。 需 要 注意 的 是 ， 由 于 没有 找到 金 钮 扣 时 的 抱怨 是 随 
机 产生 的 ， 所 以 每 次 输出 的 语句 可 能 不 一 样 。 

实话 说 ， 这 段 脚 本 模拟 寻找 金 钮 扣 的 过 程 有 些 不 太 真实 。 虽 然 每 次 执行 过 程 中 ， 输 出 
的 抱怨 都 不 一 样 ， 但 是 金 钮 扣 都 是 在 五 次 失败 后 找到 的 。 如 果 想 要 找到 金 钮 扣 的 时 间 点 更 
加 随机 ， 应 该 怎么 办 呢 ? 

解决 这 个 问题 关键 点 就 在 变量 Scounter 上 。 变 量 Scounter 的 值 是 随 着 循环 次 数 的 增加 
逐渐 由 小 变 大 的 。 由 于 变量 $counter 在 循环 中 充当 数组 Sbuttons 的 索引 ， 而 金 钮 扣 是 数组 
$buttons 中 的 第 六 个 元 素 ， 所 以 找到 金 钮 扣 的 时 间 点 就 固定 了 。 为 了 让 这 个 时 间 点 更 加 随 
机 ， 我 们 只 需要 让 变量 $counter 随机 取 值 即 可 。 

找到 了 思路 ， 再 回 过 头 去 看 看 例 7.10， 发 现 只 需要 把 第 19 行 修改 为 下 面 的 样子 : 


19 S$counter = mt rand(0,4) 7 


修改 好 后 ， 不 断 地 执行 这 个 脚本 ， 会 发 现 脚本 执行 的 结果 每 次 都 不 同 。 不 光 是 抱怨 的 
句子 不 一 样 ， 连 找到 金 钮 扣 的 时 间 点 都 变 得 随机 起 来 了 。 


7.2.3 do .… while 循环 


do.…while 循环 和 while 循环 类 似 , 它们 都 是 通过 判断 一 个 条 件 来 决定 循环 是 否 继续 的 。 
但 是 do.…while 循环 和 while 循环 不 一 样 的 地 方 ， 就 是 do.…while 会 在 判断 一 个 条 件 是 否 成 
立 之 前 就 预先 执行 一 次 需要 循环 往复 执行 的 脚本 。 它 的 结构 应 该 是 下 面 这 个 样子 : 


statement block 
} while (condition); 


特别 要 留心 condition 后 面 的 分 号 ， 这 个 是 不 能 丢掉 的 。 
我 们 来 看 一 段 脚 本 ， 体 现 一 下 do...while 循环 和 while 循环 的 区 别 。 
【 例 7.11】 do...while 循环 和 while 循环 的 区 别 。 


<?php 

2 // 计 步 器 初始 化 

| $counter = 0; 

4 

5 // 使 用 一 个 while 循环 

6 while ($counter == 1)1{ 

光 ScounteTr++7 

8 二 

9 echo "The first loop is executed for $counter times.<br>"; 
10 

ll // 计 步 器 重新 初始 化 

Ee $counter = 0; 

kk 

14 // 使 用 一 个 do. . .while 循环 

5 do 

16 皮 

和 $countert+; 

18 } while ($counter == 1); 

19 echo "The second loop is executed for $counter times."; 
0 
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在 例 7.11 这 段 脚 本 中 ， 我 们 分 别 使 用 了 一 个 while 循环 和 一 个 do...while 循环 。 在 这 
两 个 循环 中 , 我 们 均 使 用 了 变量 $counter， 用 来 记录 需要 循环 执行 的 脚本 被 执行 的 次 数 。 在 
第 一 个 脚本 中 ， 我 们 使 用 的 是 while 循环 ， 在 该 循环 中 ,循环 可 继续 的 条 件 是 变量 Scounter 
的 值 为 1, 在 每 次 循环 时 变量 Scounter 的 值 加 1; 在 第 二 个 脚本 中 , 我 们 使 用 的 是 do...while 
循环 ， 在 该 循环 时 ， 循 环 可 继续 的 条 件 依然 是 变量 Scounter 的 值 为 1， 在 每 次 循环 时 变量 
$counter 的 值 加 1 。 

现在 来 看 看 执行 结果 ， 如 图 7-8 所 示 。 


会 Favorites | 力 http//www.greatwalltea/chapter07/Ex7-11.php 全 "加 唤 ”page Safetyv Toosv 加 


The first loop is executed for 0 times. 
The second loop is executed for 2 times. 


图 7-8 do.…while 循环 和 while 循环 的 区 别 


我 们 惊奇 地 发 现 ， 第 一 段 脚 本 执行 的 次 数 为 0， 而 第 二 段 脚本 执行 的 次 数 为 2。 这 是 
为 什么 呢 ? 

先 来 看 第 一 段 肢 本， 变量 $counter 的 初始 值 为 0。 这 样 一 来 ， 循 环 继续 的 条 件 从 一 开 
始 就 不 成 立 。 因 此 变量 $counter 的 值 不 会 发 生变 化 ， 需 要 循环 执行 的 脚本 一 次 也 没有 执行 。 

而 在 第 二 段 脚 本 中 ， 虽 然 变量 Scounter 的 初始 值 也 为 0， 但 在 没有 进行 条 件 判 断 之 前 
就 已 经 加 了 1。 这 样 一 来 ， 变 量 $counter 的 值 就 变 成 了 1。 再 进行 条 件 判断 时 ， 循 环 继续 的 
条 件 成 立 ， 于 是 变量 Scounter 的 值 又 可 以 加 1， 最 后 它 的 值 就 变 成 了 2。 然 后 再 进行 条 件 判 
断 ， 这 时 条 件 不 成 立 ， 循 环 终止 。 这 样 一 来 ， 需 要 循环 执行 的 脚本 被 执行 的 次 数 就 成 了 两 
次 了 。 

如 果 把 循环 是 否 可 以 继续 的 条 件 改 成 如 下 几 种 ， 大 家 猜 猜 结果 会 是 怎样 的 呢 ? 

口 $counter< 1; 

口 $counter 一 2; 

口 $counter <= 1。 

这 里 只 告诉 大 家 ， 使 用 第 一 种 和 第 三 种 条 件 得 到 的 结果 是 一 样 的 ， 使 用 第 二 种 条 件 得 
到 的 结果 与 例 7.11 类 似 ， 但 也 许 还 是 会 出 乎 你 的 意料 。 具 体 原因 大 家 自己 分 析 一 下 吧 。 


7.2.4 ”避免 无 限 循环 


看 到 这 儿 ， 有 的 同学 可 能 就 不 淡定 了 。 为 什么 会 无 限 循环 呢 ， 不 是 都 设 定 条 件 了 吗 ? 
只 要 条 件 不 成 立 ， 循 环 就 自己 结束 了 。 哪 儿 来 的 无 限 循环 呢 ? 我 们 来 看 下 面 一 段 脚 本 。 
【 例 7.12】 无 限 循环 。 


<?php 
$fruit = array ("orange", "apple", "pear"); 
StestVar = "no"7 


AODP 


while (!StestVar == "yes"){ 


i 
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6 Skey = 0; 

7 IE ($fruit[$key] == "apple"){ 

8 StestVar = "yes"; 

号 echo "apple™; 

10 } else { 

11 echo "This $fruit[$key] is not an apple.<br>"; 
2 二 

13 Skey++7 

14 } 

i 


这 段 脚 本 无 论 在 哪个 IDE 里 都 是 不 会 报错 的 。 在 浏览 器 中 也 可 以 运行 ， 但 是 却 看 不 到 
结果 。 因 为 这 段 脚本 会 在 后 台 不 停 地 输出 一 句 话 ， 那 就 是 “The orange is not an apple.”。 
为 什么 呢 ? 

因为 我 们 把 给 变量 $key 赋值 的 那 一 句 条 件 放 在 循环 里 面 了 。 这 样 一 来 ， 每 次 循环 开始 
时 , 变量 $key 的 值 就 被 重新 设置 为 0 了 。 因 此 我 们 永远 也 等 不 到 循环 条 件 不 成 立 的 那 一 天 ， 
除非 手动 强行 终止 PHP 进程 。 因此 大 家 在 写 脚本 的 时 候 , 务必 要 小 心 在 循环 内 部 给 变量 赋 
值 。 一 招 不 慎 就 有 可 能 出 现 无 限 循 环 的 情况 。 


7.2.5 实战 练习 : 遍历 数组 的 另类 方法 


对 于 一 个 数组 而 言 ， 用 foreach 对 它 进行 遍历 ， 不 仅 可 以 获取 各 数组 元 素 的 值 ， 同 时 
也 可 以 获取 各 元 素 的 值 对 应 的 键 名 。 其实 , 我 们 也 可 以 使 用 for 循环 和 while 循环 来 过 历 一 
个 数组 。 在 本 小 节 里 , 我 们 就 来 看 看 如 何 使 用 for 循环 和 while 循环 来 分 别 遍历 一 个 数组 并 
输出 其 中 的 键 值 对 吧 。 

rE Es 
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| 使 用 FOR 和 WE 遍历 数组 


请 在 下 方 文本 框 内 输入 一 组 词 ， 使 用 半角 返 号 分 隔 相 邻 的 词 。 我 
们 将 为 您 生成 一 个 数组 ; 
CHTNa, oAPAN 


使 用 FOR 和 硅 环 剖 历数 组 


。 [0] CHINA 
0 TAPAN 


使 用 WHILE 循 环 这 历数 组 


[ol camnm 


0 maPRN 


@ intemet | Protected Mode: of 


图 7-9 使 用 FOR 和 WHILE 循环 遍历 数组 


从 图 7-9 中 可 以 看 到 ， 使 用 for 和 while 循环 输出 的 数组 结构 是 一 模 一 样 的 。 但 是 实现 
的 方法 就 大 相 径 庭 了 。 我 们 先 一 起 来 看 看 这 个 脚本 的 HTML 部 分 : 


<!DOCTYPE html> 
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<html> 
<head> 
<meta charset="UTF-8"> 
<title> 使 用 FOR 和 WHILE 遍历 数组 </title> 
<style> 
body { 
font-size:small; 
} 
.form-container{ 
width:45%; 
float:left; 
margin-right:5%; 
} 
.info-containert{ 
width:45%; 
float:left; 
margin-left:5%; 
1 
| 
font-family:monospace; 
L 
</style> 
</head> 
<body> 
<h3> 使 用 FOR 和 WHILE 遍历 数组 </h3> 
XDA 
<div class="form-container"> 
<form action="?check" method="post"> 
<div> 
请 在 下 方 文本 框 内 输入 一 组 词 ， 使 用 半角 逗号 分 隔 相 邻 的 词 。 
<textarea rows="6" cols="45" name="arrayInput"> 
</textarea> 
</div> 
<button type="submit"> 生 成 数组 </button> 
</form> 
</div> 
<div class="info-container" id="info"> /*-- 用 于 存放 输出 的 内 
容 =-*/ 
</div> 
</body> 
</html> 


在 这 个 页 面 中 使 用 了 与 上 个 实战 练习 中 相同 的 页 面 结构 ， 即 使 用 CSS+DIV 将 页 面 分 
成 了 三 个 部 分 : 一 个 用 来 存 入 页 面 的 标题 “使 用 FOR 和 WHILE 遍历 数组 ”、 一 个 用 来 存 
放 表单 、 一 个 用 来 存放 使 用 for 循环 和 while 循环 输出 的 数组 。 

在 页 面 中 定义 的 表单 ， 其 “action” 属 性 依然 是 “?check”， 而 其 “method” 属 性 依然 
是 “post”。 这 将 表明 ， 下 面 使 用 PHP 脚本 ， 将 会 使 用 当前 页 面 来 处 理 用 户 提交 的 数据 。 

现在 再 来 看 看 脚本 的 PHP 部 分 : 

<?php 

if(isset($ REQUEST['check'])){ // 检 查 用 户 是 否 已 提交 脚本 
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Sarray = explode(', ',$ REQUEST['arrayInput']); // 获 取 用 户 输入 的 数组 
Smsg = "<h4> 使 用 FOR 循环 遍历 数组 </h4><ul>"'; 
if(sizeof ($array) > 1) { 


// 当 数组 元 素 的 数量 大 于 1 


时 开始 循环 输出 
for ($i=0; $i < sizeof ($array); Si++) { 
if($i == sizeof ($array)-1) { // 当 遍历 到 数组 的 最 后 一 个 
元 素 时 在 结尾 加 上 "<ul>" 
Smsg .= '<li>['.key($array).'] '.current ($array) 
有 全 /> 
} else { 
Smsg .= '<li>['.key($array).'] '.current ($array). 
eA 


// 移 动 数 组 指针 指向 下 一 个 元 素 


next ($array); 


} 


reset ($array); // 重 置 数 组 指针 

Smsg .= "<h4> 使 用 WHILE 循环 遍历 数组 </h4><ul>'; 

$a = 0; // 定义 用 于 循环 的 计 步 器 
while ($a < sizeof ($array)) { 

if($a == sizeof ($array)-1){ 

Smsg .= '<li>['.key($array).'] '.current ($array) 
ef/ 

} else { 
Smsg .= 
eB 
next ($array); 


"<li>['.key(Sarray) .'] ' .current ($array). 


} 
Sat+t+s; 


} 
} else { 
$msg = "<1i> 请 使 用 半角 逗号 分 隔 相 邻 的 词 </1i></u1l>"; 


// 若 用 户 输入 的 数组 元 素数 量 不 足 时 提示 用 户 
上 


echo '<script language="javascript">document .getElementById 
("info") .innerHTML=""' .$msg.'"</script>"'; 
Ss :| 
在 脚本 中 ， 我 们 通过 “isset($_REQUEST['check"])” 来 判断 用 户 是 否 提 交 了 表单 。 若 用 
户 提 交 了 表单 ， 则 使 用 explodeO 函 数 将 用 户 输入 的 数据 转换 成 一 个 数组 存放 在 变量 $array 
中 。 然 后 使 用 for 和 while 循环 分 别 遍 历数 组 $array 中 的 元 素 ， 并 输出 到 变量 $msg 中 。 
现在 我 们 分 别 来 看 看 使 用 for 循环 和 while 循环 遍历 数组 元 素 的 脚本 。 
(1) 使 用 for 循环 遍历 数组 元 素 : 
Smsg = '<h4> 使 用 FOR 循环 遍历 数组 </h4><ul>'7 
if(sizeof(Sarray) > 1) { // 当 数组 元 素 的 数量 大 于 1 时 开始 循环 输出 
for ($i=0; $i < sizeof($array); $i++) { 
if($i == sizeof($array)-1) {  // 当 遍历 到 数组 的 最 后 一 个 元 素 
时 在 结尾 加 上 "<ul>" 
$msg .= '<li>['.key($array).'] ' -current ($array). 
sx/Ti></al> 


} else { 
Smsg .= "<1i>['.key($array) . '] '.current ($array). 


etl 


“se 
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next ($array); // 移 动 数组 指针 指向 下 一 个 元 素 
} 

在 这 段 脚 本 中 ， 我 们 将 数组 中 的 元 素 揉 合 到 了 一 个 无 序列 表 中 。 若 用 户 在 页 面 左 侧 的 
文本 框 中 输入 的 字符 串 中 没有 使 用 半角 逗号 的 话 ， 则 “sizeof($array) > 1” 的 判断 为 false， 
这 时 ， 页 面 标题 下 方 右 侧 不 会 输出 任何 信息 。 只 有 当 “sizeof($array) > 1” 的 判断 为 true 时 ， 
也 就 是 用 户 在 页 面 左 侧 的 文本 框 中 输入 的 字符 串 中 至 少 带 有 一 个 半角 逗号 时 ， 和 输出 数组 元 
素 的 索引 和 值 到 变量 Smsg 中 。 

值得 注意 的 是 ， 由 于 在 输出 数组 $array 的 最 后 一 个 元 素 时 ， 需 要 关闭 无 序列 表 ， 所 以 
在 for 循环 中 加 入 了 判断 当前 元 素 是 否 为 数组 $array 的 最 后 一 个 元 素 的 和 felse... 语 句 。 其 
实 也 可 以 不 使 用 这 个 判断 ， 而 直接 在 遍历 完 数组 $array 后 ， 再 关闭 无 序列 表 ， 如 下 所 示 : 

Smsg = "<h4> 使 用 FOR 循环 遍历 数组 </h4><ul>"'; 


if(sizeof ($array) > 1) { // 当 数组 元 素 的 数量 大 于 1 时 开 
始 循环 输出 
for ($i=0; $i < sizeof($array); Si++) { 
Smsg .= '<1i>[" .key($array) . '] ' .current ($array) .'</1i>'; 


next ($array); // 移 动 数 组 指针 指向 下 一 个 元 素 


| := '</ul>'; 
} 
(2) 使 用 while 循环 遍历 数组 元 素 : 
为 了 能 够 使 用 while 循环 对 同一 数组 进行 遍历 ,我们 需要 在 使 用 while 循环 遍历 数组 前 
重 置 数组 游标 。 因 此 ， 在 使 用 while 循环 前 使 用 了 “reset($array)” 函 数 。 
在 while 循环 内 ， 遍 历数 组 的 思路 与 在 for 循环 中 类 似 ， 在 这 里 就 不 再 重复 了 。 
只 需要 关注 一 点 , 那 就 是 在 灶 合 PHP 变量 与 HTML 代码 时 , 记得 多 使 用 “.=” 操 作 符 ， 
以 缩短 字符 串 表达 式 的 长 度 ， 避 人 免 可 能 出 现 的 书写 错误 。 


7.3 习 题 


(1) 请 使 用 让..else.… 语 句 结合 date0 函 数 编 写 一 段 脚 本 ， 用 来 判断 当前 的 季节 ( 春 、 
夏 、 秋 和 冬 ) 。 

(2) 请 使 用 switch 语句 结合 date0 函 数 编写 一 段 脚 本 ， 用 来 判断 当前 时 段 ( 早 、 中 
和 晚 ) 。 

(3) 请 使 用 for 循环 输出 2013 年 10 月 15 日 到 2013 年 11 月 15 日 之 间 的 所 有 日 期 。 

(4) 请 使 用 while 循环 输出 1 到 10 之 间 ( 含 1 和 10) 所 有 整数 的 和 。 

(5) 请 使 用 do.…while 循环 输出 1 到 10 之 间 ( 含 1 和 10) 所 有 整数 的 和 。 


下 和 5 二 
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在 本 章 里 ， 我 们 将 接触 函数 、 类 和 对 象 这 几 个 概念 ， 它 们 都 涉及 到 脚本 的 重用 。 那 么 
什么 是 脚本 的 重用 呢 ? 通俗 地 说 ， 脚 本 的 重用 就 是 把 需要 反复 使 用 的 脚本 单独 提取 出 来 做 
成 一 个 独立 的 函数 或 类 , 然后 在 需要 使 用 到 的 地 方 使 用 这 个 函数 或 建立 属于 这 个 类 的 对 象 。 

打 个 通俗 的 比方 ， 交 警 同志 们 在 指挥 交通 时 做 的 那 一 套 动作 就 可 以 被 定义 成 一 个 类 ， 
我 们 可 以 叫 它 “ 交 警 类 ”。 在 这 个 类 里 ， 可 以 定义 如 下 8 个 函数 : 

“ 变 道 0” 函 数 用 来 指挥 车 辆 变换 行驶 道路 ; 
“减速 慢 行 0O” 函 数 用 来 指挥 车 辆 减速 ; 
“靠边 停车 ()” 函 数 用 来 指挥 车 辆 靠边 停车 
“停车 0” 函 数 用 来 指挥 车 辆 停车 ; 

“ 右 转 0” 函 数 用 来 指挥 车 辆 右 转弯 ; 
“直行 0” 函 数 用 来 指挥 车 辆 沿 直 线 行驶 ; 

“ 左 待 转 0” 函 数 用 来 指挥 车 辆 左 转弯 待 转 ， 
“ 左 转 0” 函 数 用 来 指挥 车 辆 左 转弯 。 

在 定义 完成 这 些 函 数 后 ， 我 们 就 可 以 在 需要 使 用 相应 动作 的 地 方 直接 建立 一 个 “交警 
类 ”的 对 象 ， 然 后 通过 这 个 对 象 使 用 这 些 函数 。 


OOOOOOODD 


8.1 自 定 义 函 数 


在 前 面 的 章节 里 ， 我 们 见 过 了 大 大 小 小 的 函数 也 有 不 少 了 。 虽 然 它 们 功能 各 不 相同 ， 
不 过 都 有 一 个 共同 的 特点 , 那 就 是 它们 都 是 PHP 的 预 置 函 数 。 使 用 预 置 函数 的 好 处 就 在 于 ， 
可 以 很 方便 地 实现 某 些 功能 ， 而 不 用 自己 去 写 了 。 比 如 在 使 用 取 绝对 值 的 “absO 〇 ”函数 时 ， 
只 需要 指定 一 个 取 绝对 值 的 数 就 成 了 ， 至 于 怎么 计算 已 经 在 这 个 函数 里 写 好 了 ， 我 们 就 不 
必 操 心 了 。 

这 种 可 以 简化 脚本 的 方法 ， 我 们 也 可 以 借鉴 一 下 。 这 就 是 自 定义 函数 的 源 起 。 


8.1.1 小 试 牛刀 


本 小 节 将 试 着 编写 两 个 自 定义 函数 ， 来 体验 一 下 自 定 义 函 数 给 我 们 的 编码 带 来 的 便捷 。 

比如 ， 我 们 经 常会 对 某 些 文字 进行 加 粗 。 在 HIML 里 ， 最 常见 的 给 文字 加 粗 的 方法 就 
是 在 需要 加 粗 的 文字 两 端 加 上 “<b></b>” 标 记 对 。 为 了 避免 重复 添加 这 个 标记 对 ， 可 以 
编写 一 段 脚本 来 简化 给 文字 加 粗 的 过 程 。 

【 例 8.1】 文字 加 粗 函 数 的 使 用 。 
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<?php 
function bold($string){ 
echo "<b>".$string."</b>"; 
FE 


echo "This is not bold.<br>"; 

bold("This is bold."); 

echo "<br>Again, this is not bold"; 
?> 


入 
加 
3 
4 
5 
6 
| 
8 
9 
在 这 段 脚本 开始 ， 我 们 就 使 用 function 关键 字 定 义 了 一 个 名 为 bold 的 函数 ， 并 为 这 个 
bold0 函 数 指定 了 一 个 字符 串 类 型 的 参数 。 这 个 函数 内 部 需要 重用 的 语句 只 有 一 句 ， 作用 是 
在 变量 8string 前 后 加 上 “<b>” 和 “</b>” 标 签 ， 然 后 把 这 个 连接 起 来 的 字符 串 打印 出 来 。 
在 接 下 来 的 第 6 行 和 第 8 行 里 ， 使 用 了 echo 关键 宇 ， 打 印 不 带 文字 加 粗 标记 对 的 字符 串 ， 
而 在 第 7 行 里 ， 使 用 了 自 定义 的 bold 函数 ， 并 将 变量 Sstring 替换 成 了 需要 加 粗 的 字符 趾 。 
这 段 脚本 的 运行 结果 如 图 8-1 所 示 。 


EE 


Re emtp oo greatwalltea -apteos ES php 


宽 Favorites 。 和 莘 ntpV/www greatwallteaychapten08/BS-0Lphp 丛 " 园 "局 辆 ”page sfebyv Tosv 和 


MN This is not bold. 
This is bold. 
Again，this is not bold 


图 8-1 文字 加 粗 函 数 


在 显示 的 结果 中 , 第 二 句 话 显然 被 加 粗 了 , 而 第 一 句 和 第 三 句 都 没有 。 这 样 看 来 , bold0 
函数 的 编写 是 正确 无 误 的 ， 重 用 代码 的 目的 也 达到 了 。 之 后 ， 我 们 可 以 在 脚本 的 其 他 需要 
加 粗 文字 的 地 方 重复 地 使 用 这 个 函数 。 

现在 来 总 结 一 下 如 何 自 定义 一 个 函数 。 

定义 一 个 自 定义 函数 需要 使 用 function 关键 字 ， 然 后 指定 自 定义 函数 的 名 字 和 函数 运 
算 需 要 的 变量 ， 最 后 将 需要 重复 使 用 的 脚本 包含 在 函数 中 。 也 就 是 下 面 这 个 样子 


function functionName (parameterl, parameter2, ..) 


statement block 


} 

自 定义 函数 除了 像 例 8.1 中 的 bold0 函 数 一 样 可 以 执行 某 个 动作 之 外 ,还 可 以 返回 一 个 
值 供 其 他 函数 使 用 和 处 理 。 

在 编写 HTML 文件 时 ， 某 些 标题 需要 使 用 一 号 标题 样式 (“<h1></h1>”) ， 而 有 些 
标题 需要 使 用 二 号 标题 样式 (“<h2></h2>”) 。 重 复 的 添加 这 些 标签 十 分 的 繁琐 。 为 此 ， 
我 们 可 以 编写 一 个 headings0 函 数 ， 为 不 同 层级 的 标题 应 用 不 同 的 样式 。 

【 例 8.2】 标题 样式 函数 的 使 用 。 


于 <?php 
function headings ($string, S$level){ 
switch ($level) { 
case TL: 
$string = "<hl>$string</h1l>"; 
break; 
CS 2 


OANAON 


a 
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8 $string = "<h2>$string</h2>"; 

9 break; 

10 case 3: 

11 $string = "<h3>$string</h3>"; 

12 break; 

13 default: 

14 $string = "<p><b>$string</b></p>"; 

15 break; 

16 } 

A return($string); 

18 } 

9 

20 StestStringl = headings ("This is a second-level heading.", 2); 
1 StestString2 = headings ("This is a string without any style.", 0); 
区 2 echo $testStringl; 

| echo StestString27 

4 2> 


这 段 脚 本 定义 的 headingsO 函 数 ， 一 共 带 有 两 个 参数 : 变量 $string 和 变量 $level。 在 这 
个 自 定义 函数 里 ， 使 用 了 一 个 switch 语句 来 检测 变量 $level 的 值 。 若 变量 $level 的 值 为 1， 
则 在 变量 $string 的 前 后 加 上 “<hl1>” 和 “</h1>” 标 签 ， 若 变量 $level 的 值 为 2， 则 在 变量 
$string 的 前 后 加 上 “<h2>” 和 “</h2>” 标 签 ， 依 次 类 推 。headings0 〇 函数 只 定义 了 三 个 层 
级 的 标题 ， 因 此 我 们 在 switch 语句 中 使 用 了 default 关键 字 ， 用 来 应 对 当 变 量 $level 的 值 不 
在 1 到 3 之 间 的 情况 。 在 函数 的 最 后 ， 使 用 了 retum() 函 数 返 回 变 量 $string 的 值 。 

在 函数 外 ， 定 义 了 两 个 变量 $testStringl 和 $testString2 (假设 变量 StestStringl 是 个 二 级 
标题 ， 而 变量 $testString2 是 段 普通 文本 ) ， 然 后 使 用 headings0) 函 数 对 它们 赋值 。 最 后 使 
用 echo 关键 字 输 出 这 两 个 变量 。 

这 段 脚 本 的 运行 结果 如 图 8-2 所 示 。 


全 全 


会 Favorites 者 htp//www.greatwalltea/chapter08/Ex8-02.php 偷 " 国 "本 坊 > Pger safey” Tooks> 和 ~ 


| This is a second-level heading. 


| 


This is a string without any style. 


图 8-2 标题 样式 函数 


8.1.2 ”参数 与 返回 值 


上 一 个 小 节 定 义 了 两 个 函数 bold0 和 headings0， 这 两 个 自 定义 函数 都 带 有 参数 ， 而 且 
headings(O) 函 数 还 返回 了 一 个 值 供 其 他 语句 处 理 。 在 本 小 节 里 ， 我 们 就 来 仔细 研究 一 下 参数 
与 返回 值 的 变量 类 型 。 

现在 先 回 过 头 去 看 看 例 8.2, 脚本 里 的 两 个 参数 $string 和 $level 到 底 算 是 什么 类 型 的 变 
量 呢 ? 实际 上 ，PHP 允许 向 变量 传递 任何 类 型 的 值 ， 而 且 参 数 变量 的 类 型 是 由 传递 给 它 的 
值 决定 的 。 在 例 8.2 的 脚本 中 ， 变量 $string 应 该 是 字符 串 类 型 的 变量 ,而 $level 应 该 是 个 整 
型 变量 。 因 为 在 函数 定义 的 过 程 中 ， 我 们 就 是 把 $string 当成 字符 串 ， 而 把 $level 当成 整数 
在 使 用 的 。 


.154 
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再 来 看 一 个 例子 。 
【 例 8.3】 除法 的 使 用 。 

1 <?php 

有 function divide (Sa,Sb)1{ 
3 return ($a/$b); 

4 } 

区 

6 $c = divide (4, 2); 

时 Var dump ($c); 

8 echo "<br>"; 

泊 $c = divide (3, 2); 

10 Var dump ($c); 

让 echo "<br>"; 

E22 $c = divide (4.5, 2); 
3 var dump ($c); 

i 


这 段 脚本 定义 了 一 个 divide0 函 数 ， 它 带 有 两 个 变量 ,分别 是 变量 Sa 和 变量 8b。 这 两 
个 参数 可 以 是 任何 类 型 的 变量 。 在 函数 里 ， 我 们 使 用 retmrmn0 函 数 返回 了 变量 $a 除 以 变量 
$b 的 值 ,在 函数 外 ,我 们 定义 了 变量 Sc, 然后 使 用 divide0 函 数 为 其 赋值 ,之 后 使 用 var_dump0) 
函数 输出 变量 $c 的 值 和 变量 类 型 。 

这 段 脚 本 运行 的 结果 如 图 8-3 所 示 。 


nttpV/www.gre | 
24 . DIE 
宽 Favorites 。 莉 http://www.greatwalltea/chapter08/Ex8-03.php 从 -> ~ 口 村 > Pagev safeyv Toosv 加 -| 
int (2) 
float (1.5) 


float (2. 25) 


图 8-3 ”除法 函数 


在 脚本 中 ， 我 们 三 次 使 用 divide0 函 数 为 变量 Sc 赋值 并 输出 其 值 和 变量 类 型 ， 发 现 三 
次 结果 有 些 差 异 : 

口 第 一 次 输出 的 是 “4 除 以 2” 的 值 ， 为 整 型 变量 。 

口 第 二 次 输出 的 是 “3 除 以 2” 的 值 ， 为 浮 点 型 变量 。 

口 第 三 次 输出 的 是 “4.5 除 以 2” 的 值 ， 为 浮 点 型 变量 。 

这 样 看 来 ， 变 量 的 类 型 的 确 是 由 赋 给 它 的 值 的 类 型 来 决定 的 。 

我 们 知道 除法 中 的 被 除数 不 应 该 为 0， 否 则 除 式 将 无 法 计算 。 但 是 定义 的 divide0 函 数 
似乎 没有 采取 任何 可 靠 的 措施 来 避免 产生 无 法 计算 的 情况 。 下面 将 divide0 函 数 修改 成 如 下 
的 样子 ， 然 后 再 把 divide(5,0) 赋 值 给 变量 $c: 


function divide($a, $b){ 
Af (Sb == OT 
return FALSE; 
} else { 
return ($a/$b); 
} 


a 
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$c = divide(5,0); 
var dump ($c); 


这 样 一 来 ， 当 参数 $b 为 0 时， 系统 返回 FALSE。 那 么 变量 $c 就 成 了 一 个 布尔 型 变量 ， 
它 的 值 为 FALSE。 


8.1.3 局 部 变量 、 全 局 变量 和 静态 变量 


我 们 知道 了 所 谓 的 参数 其 实 就 是 变量 。 不 过 ， 这 种 变量 与 普通 变量 不 同 ， 它 常常 被 叫 
做 局 部 变量 ， 只 在 函数 内 部 起 作用 。 与 之 相对 的 ， 就 是 全 局 变量 了 。 所 谓 全 局 变量 ， 顾 名 
思 义 ， 其 实 就 是 在 函数 内 部 定义 的 ， 在 脚本 中 的 任何 地 方 都 有 效 的 变量 。 

至 于 静态 变量 ， 就 有 些 不 太 好 理解 了 。 变 量 之 所 以 叫 变 量 就 因为 它 是 活动 的 ， 可 以 变 
化 的 。 静 态 的 变量 , 听 上 去 就 有 些 不 可 思议 ， 一 个 可 以 变化 的 东西 怎么 能 叫 静 态 呢 ? 其 实 ， 
静态 变量 翻译 成 英语 为 static variable。 在 英语 里 ， 如 果 我 们 说 一 个 东西 是 static， 就 是 说 它 
在 任何 环境 里 都 不 会 动 或 者 不 会 改变 。 而 variable 本 身 却 是 可 以 改变 的 。 那 么 用 static 来 修 
饰 variable 就 是 说 ， 静 态 变 量 的 值 在 相应 的 函数 中 一 旦 被 定义 ， 在 下 次 使 用 该 函数 时 静态 
变量 的 值 应 该 为 上 次 修改 后 的 值 。 它 的 值 在 同一 个 作用 域 中 都 会 保持 不 变 。 

例 8.4 定义 了 三 个 函数 。 在 这 三 个 函数 中 ， 我 们 分 别 使 用 了 局 部 变量 、 全 局 变量 和 静 
态 变 量 。 来 看 看 这 些 函数 返回 的 值 有 什么 差别 吧 。 

【 例 8.4】 局 部 变量 、 全 局 变量 和 静态 变量 的 使 用 。 


1 <?php 


// 函 数 testLocal () 中 的 变量 $templ 为 局 部 变量 
3 function testLocal(){ 

4 Stemp1l = 5; 

二 $templ = Stempl * 27 

6 return (Stemp1) 

8 

9 testLocal () : 

10 echo "\$templ is ".$templ; 

lt echo "<br>"; 

testLocal (); 

i $secondRun = testLocal (); 

14 echo "\$templ is ".$secondRun; 
15 echo "<br>"; 

16 

py // 函数 testGlobal () 中 的 变量 Stemp2 为 全 局 变量 
18 function testGlobal(){ 

19 global $temp2; 

20 Stemp2 = 5; 

21 Stemp2 = Stemp2 * 2; 

22 return ($temp2); 

23 } 

24 

25 testGlobal (); 

26 echo "\$temp2 is " -Stemp27 

多 六 echo "<br>"7 

28 testGlobal (); 

29 $secondRun = testGlobal (); 

30 echo "\$temp2 is ".$secondRun; 
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当下 ECHO 有 > 

32 

Sa // 函 数 testStatic() 中 的 变量 Stemp3 为 静态 变量 
34 function testStatic()1{ 

35 static Stemp3 = 5 

36 Stemp3 = Stemp3 * 2; 

37 return ($temp3); 

38 } 

39 

40 testSstatic(); 

41 echo "\$temp3 is ".$temp3; 

42 echo "<br>"; 

43 $secondRun = testSstatic(); 

44 echo "\$temp3 is ".$secondRun; 
45 echo "<br>"; 

46 ?> 


在 这 段 长 达 46 行 的 脚本 中 ， 定 义 了 三 个 函数 ， 它 们 分 别 是 testLocal0 、testGlobal0 和 
testStatic()。 在 这 三 个 函数 里 ， 分 别 定义 了 三 个 变量 : $Stempl1、S$temp2 和 S$temp3。 随 后 分 
别 将 这 三 个 变量 与 2 的 乘积 赋 给 了 它们 自己 。 然 后 ， 每 个 函数 运行 两 次 ， 第 一 次 运行 后 ， 
输出 当前 函数 对 应 的 变量 的 值 ， 第 二 次 运行 后 ， 输 出 当前 函数 的 返回 值 。 

这 段 脚 本 运行 的 结果 如 图 8-4 所 示 。 


httpV/wwwgreai 


9 -DIETETTXTIIEITT 国 中 四 EC 


| 宽 Favorites 。 菩 httpy/www.greatwalltea/chapter08/EB8-04.php 从 ~ ”加 财 v Pagev Safeyv Toosv 各 


Notice: Undefined variable: templ in C:\greatwall\chapter08\Ex8-04.php on line 10 
| $templ is 
川 $templ is 10 
| Stemp2 is 10 

$temp2 is 10 
| Notice: Undefined variable: temp3 in C:\greatwall\chapter08\Ex8-04.php on line 41 
| Stemps is 
| $temp3 is 20 


图 8-4 ”局 部 变量 、 全 局 变量 和 静态 变量 


我 们 知道 ， 函 数 testLocal0 中 定义 的 变量 $templ 为 局 部 变量 ， 它 只 在 这 个 函数 内 部 有 
效 ， 当 在 函数 外 部 使 用 这 个 变量 时 ， 系 统 会 提示 这 个 变量 不 存在 。 我 们 只 能 通过 函数 
testLocal() 返 回 的 值 来 得 到 变量 Stemp1l 的 值 。 因 此 ， 第 一 次 运行 testLocal0 函 数 后 ， 输 出 的 
内 容 为 “templ is ”， 而 第 二 次 运行 testLocal0 函 数 后 ， 输 出 的 内 容 为 “templ is 10”。 

在 第 二 个 段落 里 ,函数 testGlobal0 中 定义 的 变量 $temp2 为 全 局 变量 ， 因 为 在 定义 变量 
S$temp2 时 使 用 了 global 关键 字 。 值 得 注意 的 是 ， 在 定义 全 局 变量 时 不 能 同时 给 变量 赋值 。 
因此 在 testGlobal0 函 数 中 另 起 了 一 行 对 变量 $temp2 赋值 。 全 局 变量 需要 在 函数 内 定义 , 但 
在 脚本 中 的 任何 地 方 都 有 效 。 因 此， 第 一 次 运行 testLocal0 函 数 后 ， 输 出 的 内 容 为 “Stemp2 
is 10”， 而 第 二 次 运行 testLocal0 函 数 后 ， 通 过 输出 该 函数 的 返回 值得 到 的 结果 依然 为 
“S$temp2 is 10” 

在 第 三 个 段落 里 ， 函 数 testStatic0 中 定义 的 变量 Stemp3 为 静态 变量 ， 因 为 在 定义 变量 


i 
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S$temp3 时 使 用 了 static 关键 字 。 与 global 关键 字 不 同 的 是 , 在 使 用 static 定义 静态 变量 的 时 
候 ， 我 们 是 可 以 同时 给 变量 赋值 的 。 需 要 注意 的 是 ， 静 态 变 量 跟 局 部 变量 一 样 ， 都 只 在 函 
数 内 部 有 效 。 因 此 ， 在 第 一 次 运行 testStatic0 变 量 后 ， 系 统 提示 变量 $temp3 未 定义 。 而 第 
二 次 运行 testStatic0 变 量 后 ， 系 统 输出 的 结果 表明 变量 $temp3 的 值 经 过 三 次 函数 的 使 用 变 
成 了 20。 

例 8.4 这 段 脚本 和 上 面 这 三 段 话 消化 起 来 不 太 容易 ， 我 们 小 小 地 总 结 一 下 : 
口 局 部 变量 只 能 在 函数 内 定义 ， 只 在 定义 它 的 函数 内 有 效 。 
口 全 局 变量 只 能 在 函数 内 定义 ， 在 定义 它 的 函数 内 外 都 有 效 。 
口 静态 变量 只 能 在 函数 内 定义 ， 只 在 定义 它 的 函数 内 有 效 ， 其 值 一 旦 改变 ， 在 下 次 

改变 前 保持 上 次 修改 后 的 值 不 变 。 

这 三 种 变量 中 ， 只 有 静态 变量 不 太 好 理解 。 这 样 ， 再 出 个 小 题 考 考 大 家 ， 看 看 大 家 对 
静态 变量 掌握 的 如 何 。 

假若 去 掉 例 8.4 第 43 行 中 的 “$secondRun =”， 只 保留 testStatic0 函 数 本 身 。 然 后 把 
第 44 行 中 的 “$secondRun” 换 成 “StestStatic()”。 那 么 第 44 行 输出 的 结果 是 什么 呢 ? 为 
什么 呢 ? 

脑袋 转 的 快 的 同学 估计 已 经 想到 了 : 输出 的 结果 应 该 为 “$temp3 is 40”, 因为 testStatic() 
函数 又 被 运行 了 一 次 。 上 次 运行 后 变量 $temp3 的 值 已 经 变 成 20 了 。 当 再 一 次 运行 该 函数 
后 ， 变 量 $temp3 的 值 自 然 就 变 成 40 了 。 

想 通 了 吗 ? 四 


8.1.4 引用 外 部 变量 


在 定义 函数 时 ， 除 了 可 以 使 用 上 面 提 到 的 局 部 变量 、 全 局 变量 和 静态 变量 之 外 ， 还 可 
以 引用 外 部 变量 。 所 谓 外 部 变量 是 相对 于 局 部 变量 来 说 的 。 与 全 局 变量 不 同 的 是 ， 外 部 变 
量 是 在 函数 外 定义 的 ， 也 只 在 函数 外 使 用 。 

在 这 一 小 节 里 ， 我 们 先 看 一 段 脚 本 ， 然 后 来 谈 谈 如 何在 函数 中 引用 在 函数 外 部 定义 的 
【 例 8.5】 在 函数 中 引用 外 部 变量 。 
<?php 

/* doublel () 函数 内 的 变量 $varInt 是 一 个 局 部 变量 ， 

而 在 该 函数 外 部 定义 的 同名 变量 是 一 个 外 部 变量 */ 


function doublel ($varInt){ 
SvarInt = $varInt * 2; 


} 


SvarInt = 5; 
doublel ($varInt); 
echo "\SvarInt1 is ".$varInt; 


重 
六 
三 
4 
5 
6 
| 
8 
本 
10 
生出 
12 // 在 double2() 函数 内 的 变量 $varInt 是 对 在 该 函数 外 部 定义 的 同名 变量 的 引用 
3 function double(&$varInt){ 

14 $varInt = SvarInt * 2; 

15 } 

16 

eh 


SvarInt = 5; 


“8s 
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18 double2 ($varInt); 
19 echo "\S$varInt2 is "-SvarInt7 


这 段 脚 本 运行 的 结果 如 图 8-5 所 示 。 


Sa Der E 


| @ http://www.greatwalltea/chapter08/E8-05.php 从 > ”四 名 7 page safey” Toosv 加 ~ 


$varIntl is 5 
$varInt2 is 10 


图 8-5 在 函数 中 引用 外 部 变量 


在 这 段 脚本 中 ， 定 义 了 两 个 函数 ， 它 们 分 别 是 double10 和 double20。 

在 double10 函 数 中 ， 定 义 了 一 个 局 部 变量 SvarInt， 并 在 函数 中 将 它 与 2 的 乘积 赋值 给 
它 自己 。 然 后 在 double10 函 数 外 又 定义 了 一 个 同名 变量 , 并 为 其 赋值 5 接着 执行 double10 
函数 。 最 后 输出 变量 Svarint 的 值 。 根据 图 8-5 显示 的 结果 可 知 ,在 函数 外 定义 的 变量 $varInt 
的 值 未 发 生变 化 。 

在 double20 函 数 中 ,我 们 在 定义 函数 时 使 用 了 一 个 引用 变量 。 注 意 在 SvarInt 前 面 加 了 
一 个 “&” 符号 ， 表 示 引 用 。 除 此 之 外 ，double20 函 数 内 外 的 脚本 与 double10 函 数 是 一 模 
一 样 的 。 可 是 ， 根 据 图 8-5 显示 的 结果 ， 在 函数 外 定义 的 变量 $varInt 的 值 变 成 了 10， 而 这 
正 是 根据 函数 内 的 那个 表达 式 可 以 得 出 的 结果 。 可 是 为 什么 呢 ? 

所 谓 引用 ， 其 实 就 是 把 这 个 外 部 变量 的 值 代入 函数 内 进行 运算 。 当 运行 double20 函 数 
时 ， 我 们 把 外 部 变量 Svarfnt 的 值 引入 了 函数 中 ， 这 时 函数 内 的 表达 式 中 使 用 的 引用 变量 就 
相当 于 这 个 外 部 变量 本 身 。 所 以 SvarInt2 的 值 就 变 成 了 10。 

这 就 是 变量 的 引用 。 


8.1.5 函数 的 引用 


如 果 函 数 只 能 在 定义 它 的 脚本 文件 中 使 用 那 就 太 逊 色 了 .PHP 提供 了 include 和 require 
两 种 方法 用 于 在 一 个 脚本 中 引用 其 他 脚本 中 定义 的 函数 。 确 切 地 说 ， 是 引用 定义 这 些 函 数 
的 脚本 文件 。 

还 记得 在 “8.1.1 小 试 牛刀 ”中 定义 的 bold0 函 数 吗 ? 如 果 我 们 想 在 其 他 的 脚本 中 也 
使 用 bold0 函 数 ， 就 可 以 把 bold0 函 数 的 定义 脚本 写 入 一 个 叫 自 定义 文件 名 的 脚本 文件 〈 比 
如 function.inc〉 中 。 然 后 在 需要 使 用 bold0 函 数 的 脚本 文件 开始 ， 写 上 下 面 这 句 话 : 

include "function.inc"; 

注意 ， 这 时 文件 fanction.inc 必须 与 写 入 这 句 话 的 脚本 文件 在 同一 个 目录 中 ， 文 件 
fonction.inc 才能 够 被 成 功 引 用 。 在 实际 编程 中 ， 我 们 通常 会 把 被 引用 的 文件 都 放 在 一 个 叫 
inc 的 文件 夹 中 ， 然 后 使 用 相对 路 径 来 引用 它们 。 假 设 把 刚才 创建 的 fonction.inc 文件 放 入 
了 inc 文件 夹 中 ， 这 个 inc 文件 夹 与 需要 引用 fonction inc 的 脚本 文件 层级 相同 ， 那 么 可 以 
使 用 下 面 这 句 话 : 


i 
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include "../inc/function™; 


但 是 如 果 日 后 引用 了 function.inc 的 脚本 文件 换 了 地 方 ， 就 得 修改 脚本 了 。 为 了 避免 出 
现 这 种 烛 熔 的 情况 ， 也 可 以 使 用 绝对 路 径 来 引用 fonction.inc。 假 设 需 要 引用 function.inc 
的 脚本 文件 在 网 站 的 根 目 录 下 ， 而 包含 function.inc 的 inc 文件 夹 在 网 站 的 根 目录 下 的 lib 
文件 夹 中 。 我 们 可 以 使 用 下 面 这 句 话 : 


include "/lib/inc/function™; 


这 样 一 来 , 无 论 我 们 引用 了 fonction inc 的 脚本 文件 放 在 什么 地 方 ， 都 不 用 再 修改 脚本 了 。 

一 般 来 说 ， 我 们 使 用 require 的 时 候 比 使 用 include 的 时 候 多 。 因 为 一 旦 被 引用 的 脚本 
有 错误 ,使 用 include 会 导致 系统 警告 ,而 使 用 require 则 可 以 让 系统 直接 终止 被 引用 脚本 ， 
不 发 出 任何 警告 。 

还 有 ， 我 们 可 以 在 一 个 脚本 中 引用 多 个 文件 ， 也 可 以 在 被 其 他 脚本 文件 引用 的 文件 中 
再 引用 其 他 的 文件 。 这 样 虽然 很 方便 ， 但 可 能 造成 某 些 文件 被 mclude 或 require 了 多 次 。 
为 了 解决 这 个 问题 ，PHP 还 提供 了 include_once 或 require_once 这 样 两 条 语句 。 使 用 它们 ， 
就 不 怕 重 复 引 用 了 。 


8.2 类 


在 本 章 的 开始 ， 我 们 用 交通 警察 打 了 个 比方 ， 说 明了 函数 、 类 和 对 象 之 间 的 关系 。 上 
一 节 ， 系 统 的 了 解 了 什么 是 函数 。 本 节 ， 我 们 就 好 好 地 讲解 讲解 什么 是 类 、 什 么 是 对 象 。 

在 PHP 中 ,所 谓 “ 类 ”， 指 的 是 一 个 包含 一 些 数据 和 如 何 访问 与 修改 这 些 数据 的 方法 
的 包 。 这 些 数据 就 是 一 些 变量 ， 不 过 在 类 里 ， 它 们 被 称 做 “属性 ”。 而 访问 与 修改 这 些 数 
据 的 方法 就 是 一 些 函 数 ， 只 是 在 类 里 才 会 被 叫做 “方法 ”。 

当 定 义 好 一 个 类 之 后 ， 并 不 能 直接 读 取 和 使 用 类 中 定义 的 “属性 ”和 “方法 ”， 而 是 
要 通过 创建 一 个 基于 这 个 类 的 对 象 来 使 用 它 。 这 个 创建 对 象 的 过 程 就 叫做 “实例 化 ”。 读 
者 可 以 把 类 想像 成 一 个 模具 ， 而 把 对 象 想象 成 用 这 个 模具 做 出 来 的 复制 品 。 一 个 模具 可 以 
用 来 做 成 千 上 万 个 复制 品 ， 而 一 个 类 也 可 以 被 实例 化 成 无 数 个 对 象 。 

如 果 写 了 一 个 类 ， 你 自然 是 知道 类 的 各 种 方法 是 如 何 实现 的 。 但 是 这 并 不 意味 着 你 必 
须知 道 一 个 类 中 定义 的 各 种 方法 是 如 何 实现 的 ， 才 可 以 使 用 这 个 类 。 因 此 ， 你 可 以 在 你 的 
脚本 中 使 用 已 经 定义 好 的 类 来 创建 对 象 ， 进 而 使 用 类 中 定义 的 属性 和 方法 。 


8.2.1 如 何 定义 类 


例 8.6 定义 了 一 个 叫 ArithmeticO 的 类 。 我 们 一 起 来 看 看 这 个 类 是 怎么 写 的 。 
【 例 8.6】 简单 算术 类 的 使 用 。 
<?php 

class Arithmetic { 


// 定 义 两 个 成 员 变 量 


var $a = 0; 
var Sb = 0; 


[a 


*160°* 
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/定义 一 个 让 两 成 员 变 量 相 加 的 方法 

8 function addUp(){ 

9 return S$this -> a + $this -> b; 
10 i 

1 

12 // 定 义 一 个 让 两 成 员 变 量 相 减 的 方法 

| function minusDown (){ 

14 return $this -> a - $this -> b; 
15 } 

16 

fi // 定 义 一 个 让 两 成 员 变量 相 乘 的 方法 

18 function timesUp(){ 

19 return $this -> a * $this -> b; 
20 1 

2 

22 // 定 义 一 个 让 两 成 员 变 量 相 除 的 方法 

多 人 function divideDown(){ 

24 return S$this -> a / $this -> b; 
25 

26 | 

27 

28 // 定 义 一 个 基于 Arithemtic 类 的 对 象 SdoMath 
29 $doMath = new Arithmetic; 

30 

31 // 为 对 象 $doMath 的 两 个 属性 赋值 

32 $doMath -> a = 6; 

33 S$doMath -> b = 3; 

34 

35 // 打 印 一 些 结果 

36 echo "6 + 3 = ".$doMath -> addUp(); 

37 echo "<br>"; 

38 echo "6 - 3 = ".S$doMath -> minusDown(); 
3 echo "<br>"; 

40 echo "6 x 3 = ".$doMath -> timesUp(); 
41 echo "<br>"; 

42 echo "6 / 3 = ".$doMath -> divideDown(); 
3 


这 段 脚 本 创建 了 Arithmetic 类 ， 并 在 这 个 类 中 定义 了 两 个 成 员 变 量 和 四 个 成 员 函 数 。 
用 专业 的 角度 来 说 ， 应 该 是 在 一 个 类 中 定义 了 两 个 属性 和 四 个 方法 。 定 义 类 时 需要 使 用 关 
键 字 “class”， 就 像 下面 这 样 : 

class className { 

在 类 中 定义 成 员 变量 时 ， 前 面 得 加 上 关键 字 “var”， 这 时 可 以 选择 为 其 赋值 。 在 这 里 
赋值 的 目的 除了 让 成 员 变量 有 个 初始 值 之 外 ， 也 可 以 指定 成 员 变量 的 变量 类 型 。 比 如 在 这 
里 为 成 员 变量 $a 和 $b 赋值 0， 表 明 它 们 都 是 整 型 变量 。 

在 类 中 定义 成 员 函 数 时 ， 若 在 函数 中 需要 使 用 成 员 变 量 的 , 需要 通过 特殊 变量 “S$this” 
来 引用 。 这 个 特殊 变量 相当 于 一 个 占 位 符 ， 用 来 替代 还 没有 创建 的 对 象 。 一 旦 创建 了 基于 
这 个 类 的 某 个 对 象 ， 我 们 就 可 以 使 用 “S$objectName ->” 来 代替 “Sthis ->” 访 问 这 个 对 象 
的 属性 和 方法 了 。 

另外 ， 在 成 员 函 数 中 用 到 的 成 员 变 量 是 不 需要 在 函数 名 后 面 的 括号 中 事先 声明 的 。 这 
与 定义 普通 函数 有 些 不 同 。 假 设 ， 我 们 要 定义 一 个 普通 函数 用 来 让 两 个 数 相 加 ， 得 写成 下 


a 
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面 的 样子 : 

function addUp ($a, $b){ 

return ($a + $b); 

} 

而 在 类 中 ， 直 接 让 函数 名 后 面 的 括号 空 着 就 好 ， 除 非 你 需要 添加 其 他 的 参数 。 

在 定义 好 Arithmetic 类 之 后 ， 就 可 以 创建 基于 这 个 类 的 对 象 了 。 在 例 8.6 中 ， 我 们 使 
用 关键 字 “new” 创 建 了 一 个 基于 Arithmetic 类 的 对 象 gdoMath。 然 后 使 用 “$doMath ->” 
定义 了 该 对 象 两 个 属性 的 值 。 注 意 ， 通 过 “->” 访 问 对 象 的 属性 时 ， 属 性 前 的 “$” 一 定 要 
去 掉 。 最 后 ， 使 用 “$doMath ->” 分 别 引 用 了 类 的 四 种 方法 来 输出 两 个 属性 的 值 相 加 、 相 
减 、 相 乘 和 相 除 的 结果 。 

类 和 普通 函数 一 样 ， 可 以 与 其 他 脚本 分 离开 来 。 我 们 可 以 把 所 有 的 类 都 存放 在 一 个 独 
立 的 文件 里 ， 然 后 通过 include、require、include_once 或 require_once 来 引用 这 个 文件 ， 从 
而 使 用 文件 中 存放 的 类 。 


8.2.2 ”魔术 方法 _construct() 和 _ destruct() 


在 PHP 5 中 ， 有 一 些 变量 叫做 魔术 变量 ， 有 一 些 方法 被 叫做 魔术 方法 。 这 么 叫 不 是 因 
为 它们 会 变 魔术 , 而 是 因为 它们 可 以 很 方便 地 帮助 我 们 获取 一 些 在 之 前 的 PHP 版 本 里 很 难 
获取 的 数据 和 使 用 一 些 可 以 让 PHP 类 变 得 更 加 易 用 的 方法 。 值 得 注意 的 是 ， 所 有 的 魔术 变 
量 和 魔术 方法 前 面 都 会 有 两 道 下 划 线 (_) 。 

在 这 一 小 节 里 ,我 们 将 讨论 一 下 _construct0 和 destruct0 方 法 ， 这 两 种 方法 在 英语 里 
被 称 作 constructor 和 destructor， 其 词根 都 是 “or”， 表 明 它 们 是 用 来 做 某 件 事 情 的 人 或 工 
具 。 这 里 显然 把 它们 当成 工具 更 合适 。 注 意 ， 这 两 种 方法 〈 注 意 我 们 用 的 词 是 “方法 ”) 
只 能 用 在 PHP 类 中 ， 不 能 用 在 其 他 地 方 。 它 们 的 功能 与 PHP 类 也 有 着 密切 的 关系 : 

口 在 类 被 “实例 化 ”的 过 程 中 ，_construct() 方 法 中 的 代码 会 被 自动 执行 。 

口 在 类 被 “销毁 ”时 ，_destruct0 方 法 中 的 代码 也 会 被 自动 执行 。 

在 例 8.7 中 ， 定 义 了 一 个 “汽车 ”类 (Car) ， 然 后 为 这 个 类 定义 了 一 个 属性 “汽油 ” 
(S$gas) 和 一 个 方法 “加 油 ” (addGas()) ， 然 后 将 这 段 脚本 另存 为 文件 class.inc.php。 

本 节 例 8.7 至 例 8.14 是 一 个 完整 的 实例 ,其 中 代码 行 号 不 连续 是 因为 空 行 行 号 没 给 出 。 

【 例 8.7】 魔术 方法 _construct0 的 使 用 。 

人 <?php 

区 // 定 义 一 个 类 并 为 其 命名 Car 

3 class Car { 

4 //define a property named gas 

5 var $gas = 0; 

6 Var $addAmount = 0; 

沈 
8 


// 定 义 一 个 方法 并 为 其 命名 addGas 


9 function addGas ($amount = 0){ 

10 if (is int($amount)){ 

3 if ($amount + Sthis -> gas <= 50) { 

全 Sthis -> gas = Sthis -> gas + $amount; 

a Sthis -> addAmount = $amount; 

14 return ("$amount gallons are added to your gas tank."); 
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证 二 } else { 

16 return ($this -> gas." gallons are already in 
your gas tank.<br> 

人 You can add ".abs(50 - $this -> gas)." gallons 
more."); 

18 } 

19 } else { 

20 return "Excuse me! An integer is required."; 

21 J 

22 下 

23 

24 // 定 义 _construct () 方 法 ， 初 始 化 油箱 现存 油 量 

25 function construct(){ 

26 $this -> gas = 50; 

27 } 

28 } 

9 2> 


在 这 个 Car 类 中 ， 使 用 了 魔术 方法 _construct0， 而 且 在 里 面 为 “$this” 的 汽油 属性 定 
义 了 一 个 初始 值 50。 
然后 ， 新 建 一 个 脚本 文件 Ex8-07.php， 并 在 文件 中 写 入 如 下 所 示 的 脚本 : 


p <?php 

已 require once "class.inc.php"; 

3 

4 // 创 建 一 个 基于 car 类 的 对 象 $SfordFiesta 
5 $fordFiesta = new Car; 

6 

// 显 示 Ford Fiesta 现存 油 量 

8 echo "Gas Level: ".$fordFiesta -> gas." gallons<br><br>"; 
9 

10 // 给 Ford Fiesta 加 油 

11 echo $fordFiesta -> addGas (10); 

i 

这 段 脚本 运行 的 结果 如 图 8-6 所 示 。 
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Gas Level: 50 gallons 
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要 知道 ， 在 class.inc.php 的 第 5 行 ， 为 属性 $gas 赋 的 值 是 0， 而 在 Ex8-07.php 的 第 8 
行 ， 是 直接 输出 “SfordFiesta -> gas” 的 值 。 而 在 这 之 前 ， 我 们 没有 使 用 Car 类 的 任何 方法 
来 改变 这 个 属性 。 唯 一 的 解释 只 能 是 _construct0 方 法 中 的 代码 在 对 象 fordFiesta 被 创建 的 
同时 被 运行 了 。 在 __construct0 方 法 里 ， 我 们 使 用 了 “Sthis -> gas = 50” 这 样 的 语句 ， 而 
“S$this” 指 代 的 就 是 诸如 $fordFiesta 这 样 的 基于 Car 类 的 对 象 。 因 此 ， 属 性 $gas 的 值 才 会 由 
0 变 成 50。 
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之 后 ,在 Ex8-07.php 的 第 11 行 , 执行 了 对 象 SfordFiesta 的 addGas() 方 法 ， 并 为 其 指定 
了 一 个 参数 : 需要 添加 的 汽油 量 。 注 意 ， 在 定义 这 个 addGas() 方 法 时 ， 我 们 指定 了 参数 
$amount， 并 为 其 指定 了 一 个 默认 值 0。 这 样 一 来 ， 如 果 在 使 用 addGas() 方 法 时 ， 没 有 指定 
参数 值 ， 系 统 就 会 自动 使 用 这 个 默认 值 ， 从 而 保证 脚本 不 会 报错 。 另 外 ， 在 addGas() 方 法 
内 ， 我 们 使 用 了 两 层 站..else 语句 的 嵌 套 : 首先 判断 了 $amount 是 否 为 一 个 整数 ， 若 不 是 ， 
则 提示 输入 一 个 整数 ， 然 后 判断 了 $amount + Sthis -> gas 的 值 是 否 超过 10， 若 不 是 ， 则 提 
示 油 箱 已 满 ， 无 需 加 油 。 

大 家 可 以 尝试 一 下 ， 把 脚本 第 11 行 修改 成 如 下 的 样子 


Ll echo $fordFiesta -> addGas(); 
或 
I echo $fordFiesta -> addGas ("Hello!"); 


看 看 会 输出 什么 结果 ? 和 你 想 的 是 否 一 样 呢 ? 

与 _construct( 方 法 中 的 语句 会 在 一 个 类 被 实例 化 时 得 到 执行 一 样 , 如 果 我 们 在 类 中 定 
义 了 _destuct(0 方 法 ， 这 个 方法 中 的 语句 也 会 在 类 被 销毁 时 被 执行 。 但 与 _constructO 类 不 
同 的 是 ，_destruct() 方 法 不 允许 携带 任何 参数 。 

那么 什么 时 候 才 需要 使 用 _destruct( 方 法 呢 ? 通常 情况 下 ， 我 们 一 般 不 需要 使 用 这 个 
方法 。 不 过 如 果 你 想 在 某 个 类 被 销毁 时 做 一 些 清 扫 工 作 ， 也 可 以 加 上 它 。 另 外 ， 如 果 一 段 
脚本 需要 调试 ， 也 可 以 把 _destructO 当 成 一 个 调试 工具 。 

在 后 面 的 章节 中 ， 会 看 到 关于 _destruct0) 方 法 的 示例 。 因 此 ， 在 这 里 就 不 再 袭 述 了 。 


8.2.3 ”类 的 继承 


在 前 面 的 小 节 里 ， 我 们 讨论 了 类 与 对 象 的 关系 。 对 象 与 类 之 间 只 存在 “复制 ”关系 
对 象 是 类 的 具体 化 和 实例 化 ， 而 类 是 众多 对 象 的 抽象 化 。 和 类 与 对 象 的 关系 不 同 的 是 ， 继 
承 是 类 与 类 之 间 的 一 种 关系 。 比 如 ， 我 们 有 一 个 类 叫 Car， 在 这 个 类 中 ， 定 义 了 一 些 所 有 
汽车 都 会 有 的 通用 属性 和 方法 。 然 后 又 定义 了 一 个 类 叫 FordCar， 而 这 个 类 中 除了 拥有 所 
有 在 Car 类 中 定义 的 属性 和 方法 之 外 ， 还 新 增 了 一 些 只 有 FordCar 类 才 有 的 属性 和 方法 。 
那么 FordCar 类 和 Car 类 之 间 的 关系 就 存在 着 继承 与 被 继承 的 关系 ,被 继承 的 一 方 叫做 “ 父 
类 ”， 而 继承 的 一 方 叫做 “ 子 类 ”。 

父 类 和 子 类 间 的 关系 就 有 点 像 父 子 关系 ， 儿 子 会 继承 父亲 的 某 些 特点 ， 但 又 与 父亲 是 
完全 不 同 的 两 个 人 。 类 也 一 样 ， 父 类 有 的 属性 ， 子 类 可 能 都 会 有 ， 而 子 类 有 的 属性 ， 父 类 
却 不 一 定 有 。 

我 们 应 该 如 何 使 用 PHP 的 语言 来 表述 这 种 关系 呢 ? 先 来 看 一 段 脚本 。 

【 例 8.8】 类 的 继承 。 

还 是 先 在 class.inc.php 中 写 入 如 下 脚本 : 


30 // 定 义 一 个 继承 自 Car 的 子 类 并 为 其 命名 FordCar 
3 下 class FordCar extends Car { 

32 //Define some attributes 

33 Var S$gasConsumption = 1.6; 

34 Var SavgSpeed = 50; 
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35 

36 function timeCalc(){ 

37 $timeToDie = (Sthis -> gas / $this -> gasConsumption) / $this 
-> avgSpeed; 

38 return ("You can drive no more than $timeToDie hours."); 

39 } 

40 } 


然后 新 建 一 个 脚本 文件 Ex8-08.php， 并 写 入 如 下 脚本 : 


1 <?php 

区 require once "class.inc.php"; 
3 

4 $fordFox = new FordCar; 

5 

6 echo "Gas Level: ".$fordFox -> gas." gallons.<br><br>"; 
多 echo $fordFox -> timeCalc(); 
8 echo "<br>"; 

要 

10 echo $fordFox -> addGas (10); 
1 


在 例 8.8 的 前 一 段 脚本 中 ， 定 义 了 一 个 类 FordCar。 值 得 注意 的 是 ， 在 定义 FordCar 这 
个 类 时 , 使 用 了 extends 关键 字 , 表示 FordCar 这 个 类 是 从 Car 这 个 类 继承 来 的 。 在 FordCar 
这 个 类 中 ， 定 义 了 一 个 方法 tmeCalc(0) 来 估算 按照 现存 油 量 ， 车 还 能 走 多 久 。 在 timeCalc() 
方法 中 ， 使 用 了 在 Car 类 中 定义 的 gas 属性 。 

然后 在 创建 FordCar 类 之 后 ， 在 Ex8-08.php 这 个 文件 里 创建 了 一 个 基于 FordCar 类 的 
对 象 ， 并 为 其 命名 $fordFox。 接 着 输出 了 这 和 辆 车 的 现 有 油 量 、 预 计 可 行驶 时 间 和 执行 加 油 
的 结果 。 在 输出 这 些 信息 时 ， 我 们 使 用 了 从 Car 类 继承 过 来 的 gas 属性 和 addGas() 方 法 。 

这 段 脚 本 输出 的 结果 如 图 8-7 所 示 。 从 图 中 可 以 看 出 ， 脚 本 运行 正常 ， 没 有 报错 。 

在 这 个 例子 中 ，Car 是 FordCar 的 父 类 ,而 FordCar 是 Car 的 子 类 。FordCar 继承 了 Car 
中 定义 的 所 有 属性 和 方法 ， 并 且 在 Car 原 有 属性 和 方法 的 基础 上 添加 了 新 的 属性 和 方法 。 
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Gas Level: 50 gallons. 
You can drive no more than 0. 625 hours. 


50 gallons are already in your gas tank. 
You can add 0 gallons more. 


图 8-7 类 的 继承 


除了 全 盘 继 承 ， 还 可 以 在 子 类 中 修改 从 父 类 继承 过 来 的 属性 和 方法 ， 也 包括 对 父 类 中 
的 _construct0 方 法 的 修改 。 我 们 来 看 下 面 这 个 例子 。 

【 例 8.9】 继承 并 修改 父 类 的 属性 和 方法 。 

先 在 class.inc.php 中 定义 一 个 新 的 类 ， 并 为 其 命名 GMCar。 


42 //define another class extended from class Car 
43 class GMCar extends Car { 
44 Var S$gasType = array ("Gas 89" => 7.31, 
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"Gas 92" => 1. 01 
"as 95 => 0327 
WLS On = TT 


function gasQuot(){ 
if ($this -> addAmount > 0) { 
$returnSstring = "You need to buy ".$this -> addAmount." 
gallons of gas. 
<br>You need to pay:<ul>"; 
foreach ($this -> gasType as $type => SunitPrice) { 
SreturnString = $returnString."<1i>$".$unitPrice * 
Sthis -> addAmount. 
"” for ".$this ->addAmount." gallons of 
SEyPe: "li 
} 
$returnstring = $returnstring."</ul>"; 
return ($returnSstring); 
} else { 
return("No gas fee"); 
} 
| 


function construct($gas){ 
Sthis -> gas = $gas; 
} 
} 


接着 新 建 一 个 脚本 文件 ， 并 命名 为 Ex8-09.php。 然 后 写 入 如 下 脚本 : 


b 
区 
号 
4 
二 
6 
J 
8 


<?php 
require "class.inc.php"; 


SGMPolo = new GMCar (40); 
SGMPolo -> addGas (10); 


echo $GMPolo -> gasQuot (); 


这 段 脚本 的 运行 结果 如 图 8-8 所 示 。 


jp 
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You need to buy 10 gallons of gas. 
You need to pay: 


es。 $73.1 for 10 gallons of Gas 89. 
。$78.1 for 10 gallons of Gas 92. 
。 $83.2 for 10 gallons of Gas 95. 
es。 $77.8 for 10 gallons of Disl 0. 


图 8-8 继承 并 修改 父 类 的 属性 和 方法 


在 例 8.9 所 示 中 ， 我 们 重新 定义 了 属性 gas 的 初始 值 ， 而 这 个 属性 是 GMCar 类 从 Car 
类 中 继承 的 。 在 继承 过 程 中 ， 我 们 可 以 通过 重新 定义 属性 和 方法 来 使 这 种 继承 关系 发 生 某 


些 变化 。 


"66" 


T 


这 与 儿女 与 父母 之 间 的 继承 关系 类 似 ， 有 部 分 基因 会 在 继承 过 程 中 发 生变 异 。 
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8.2.4 类 的 私有 元 素 


与 函数 类 似 ， 我 们 可 以 在 类 中 定义 属于 该 类 私有 的 属性 和 私有 方法 。 通 过 定义 私有 属 
性 和 方法 ， 就 可 以 隐藏 类 的 某 些 属性 和 方法 ， 一 定 程度 上 保证 了 这 些 被 隐藏 的 属性 和 方法 
不 会 在 使 用 过 程 中 被 算 改 。 
为 了 说 明 这 个 问题 ， 还 是 先 回 过 头 去 看 看 例 8.7 中 的 Car 类 ， 所 有 的 属性 和 变量 都 是 
可 以 在 基于 该 类 的 对 象 中 访问 的 。 也 就 是 说 使 用 这 个 Car 类 , 所 有 的 属性 都 有 可 能 被 修改 
所 有 的 方法 都 有 可 能 被 使 用 。 如 果 我 们 想 隐 藏 某 些 属性 和 方法 ， 可 以 定义 这 些 属性 和 方法 
时 使 用 private 关键 字 ， 就 像 例 8.10 所 示 中 一 样 。 

【 例 8.10】 类 的 私有 属性 。 


72 class UnitCounter { 


373 Private Sunits = 0; 

74 private S$weightPerUnit = 1.0; 
Yi 

76 function numberOfUnit (){ 

I return S$this -> units; 

78 1 

TQ 

80 function addUnit (Sn = 1)1{ 

81 Sthis => units = $this => units:+ $ns 
82 } 

830 °F 


我 们 在 class.inc.php 中 输入 如 上 所 示 的 这 段 脚 本 。 在 这 段 脚本 中 ， 定 义 了 一 个 名 为 
UnitCounter 的 类 ， 这 个 类 是 用 来 计数 的 。 在 这 个 类 中 ， 定 义 了 两 个 私有 属性 : 一 个 是 “ 物 
品 个 数 (units) ”， 另 一 个 是 “物品 单 重 (weightPerUnit) ”。 这 两 个 私有 属性 是 不 可 以 
在 对 象 中 直接 访问 的 。 然 后 定义 了 两 种 方法 : 一 是 “物品 总 数 CnumberOfUnit) ”， 另 一 
个 是 “增加 物品 (addUnit) ”。 只 有 使 用 这 两 种 方法 才 可 以 访问 上 面 的 两 个 私有 属性 。 

接着 新 建 Ex8-10.php 文件 ， 并 在 文件 中 输入 以 下 脚本 : 


和 <?php 

慰 require _ once "class.inc.php"; 
3 

4 Sapple = new UnitCounter; 

号 

6 // 下 面 两 行 脚 本 会 报错 

echo S$apple -> unit; 

8 echo S$apple -> weightOofUnit; 
9 

WO // 下 面 两 行 脚 本 会 正常 执行 

有 $apple -> addUnit (10) 

2 echo $apple -> numberOfUnit(); 
EE ER 


在 这 段 脚 本 中 ， 定 义 了 一 个 基于 UnitCounter 类 的 对 象 Sapple， 然 后 使 用 $apple 对 象 来 
访问 unit 和 weightOfUnit 属性 。 接 着 又 使 用 了 addUnit 方法 增加 了 10 只 苹果 , 然后 输出 现 
有 苹果 的 数量 。 正 如 脚本 中 备注 说 的 一 样 ， 这 段 脚本 的 运行 结果 如 图 8-9 所 示 。 

通过 输出 结果 ， 我 们 发 现 系 统 认 定 这 两 个 属性 没有 定义 。 这 也 说 明 ， 我 们 可 以 通过 定 
义 私 有 属性 来 隐藏 某 些 不 想 让 类 的 使 用 者 修改 的 内 容 。 
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Notice: Undefined property: UnitCounter::$unit in C:\greatwall\chapter08\Ex8-10.php on 
line 7 


Notice: Undefined property: lnitCounter::$weightOflnit in C:\greatwall\chapter08\Ex8— 


10.php on line 8 
10 


图 8-9 类 的 私有 属性 


私有 方法 与 私有 属性 是 在 私密 性 方面 是 一 样 的 。 再 来 看 一 段 脚本 。 例 8.11 这 段 脚本 定 
义 了 一 个 FreightCalculator 类 ， 在 这 个 类 中 ， 定 义 了 两 个 私有 变量 : $numberOfCases 和 
S$totalWeight。 然 后 又 定义 了 两 个 私有 方法 : KgTotal0 和 CaseTotal()。 

【 例 8.11】 类 的 私有 方法 。 

我 们 还 是 在 class.inc.php 中 定义 FreightCalculator 类 。 

87 class FreightCalculator { 


88 private $numberOfCases; 

89 private $totalWeight; 

90 

91 function totalFreight (){ 

92 return round($this -> CaseTotal() + Sthis -> KgTotal()); 

93 

94 

95 private function CaseTotal () 1{ 

96 if($this -> numberOfCcases > 5)1{ 

97 return 5; 

98 } else { 

99 return S$this -> numberOfCases * 1.0; 

100 } 

101 } 

102 

103 private function KgTotal(){ 

104 if (Sthis -> numberofCases > 5){ 

105 return ((Sthis -> totalWeight / Sthis -> numberOfCases) * 
(Sthis -> numberOfCases - 2)) 

106 人 

107 } else { 

108 return 0; 

109 it 

110 } 

了 

站 地 function construct ($numberOfCases, S$totalWeight){ 

dls $this -> numberOfCases = $numberOfCases; 

114 Sthis -> totalWeight = $totalWeight; 

ds 上 

GO 


在 这 个 FreightCalculator 类 中 ， 我 们 的 计算 公式 是 : 若 件 数 CnumberOfcases) 小 于 5， 
则 只 按 件 计价 ， 每 件 10 元 ; 车 件数 (numberOfCases) 大 于 5， 则 除去 按 件 计价 的 部 分 后 ， 
按 重 计价 。 最 后 输出 总 运费 。 

接 下 来 ， 在 新 建 的 Ex8-11 php 中 输入 如 下 脚本 : 
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于 <?php 

全 require once "class.inc.php"; 

3 

4 SexpressFreight = new FreightCalculator (15，200) 
5 

6 // 这 名 脚本 可 以 运行 

了 echo "You need to pay $".$expressFreight -> totalFreight()."."; 
8 

9 // 这 句 脚本 会 报错 

10 echo $expressFreight -> CaseTotal() 

> 


Ex8-11.php 定义 了 一 个 对 象 SexpressFreight， 同 时 指定 了 “件数 CnumberOfCases) ” 
为 15，“ 总 重量 (totalWeight) ”为 200。 最 后 输出 总 运费 和 总 件数 。 

这 段 脚 本 的 运行 结果 如 图 8-10 所 示 。 

通过 截图 可 知 ， 总 运费 正常 输出 ， 而 总 件数 则 输出 异常 。 系 统 提 示 无 法 调用 对 象 的 私 
有 方法 。 
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You need to pay $40. 
Fatal error: Call to private method FreightCalculator::CaseTotal() from context ’’ in 
C:\greatwall\chapter08\Ex8-11.php on line 10 


图 8-10 类 的 私有 方法 


8.2.5 ”类 的 静态 元 素 


在 类 中 ， 也 可 以 把 属性 和 方法 定义 为 静态 的 。 所 谓 的 静态 属性 与 普通 属性 不 同 之 处 在 
于 ， 普 通 属性 的 值 在 每 个 基于 某 类 的 对 象 中 是 相互 独立 的 ， 而 静态 属性 的 值 在 每 个 基于 某 
类 的 对 象 中 是 共享 的 。 也 就 是 说 ， 静 态 属性 一 旦 在 一 个 基于 某 类 的 对 象 中 被 改变 了 ， 那 么 
在 所 有 基于 该 类 的 对 象 中 ， 这 个 静态 属性 的 值 也 就 都 跟着 改变 了 。 

例 8.12 定义 了 一 个 Revenue 类 , 用 于 统计 一 家 商店 某 日 的 销售 的 产品 类 型 及 每 类 商品 
的 销售 额 。 这 两 个 数据 被 存放 在 两 个 私有 属性 ScommodityType 和 $amount 中 。Revenue 类 
同时 还 统计 所 有 商品 的 销售 总 额 和 总 计 销 售 了 多 少 种 商品 ， 这 两 个 数据 则 被 存放 在 了 两 个 
静态 属性 $totalRevenue 和 $numberOfTypes 中 了 。 值得 注意 的 是 ， 所 有 的 基于 Revenue 类 的 
对 象 都 可 以 访问 这 两 个 静态 属性 并 修改 它们 的 值 ， 但 是 与 普通 属性 不 同 的 是 ， 在 使 用 静态 
属性 时 ， 需 要 使 用 类 名 加 双 冒 号 (如 Revenue::) 而 不 是 之 间 使 用 $this 加 箭头 符号 〈 如 
S$this->) 。 此 时 ， 我 们 可 以 把 这 个 类 看 成 是 个 常量 。 

一 起 来 看 看 这 段 脚本 。 

【 例 8.12】 静态 属性 的 使 用 

我 们 先 在 class.inc.php 中 定义 Revenue 类 ， 脚 本 如 下 : 


120 class Revenue { 
2 private S$unitPrice; 
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122 private $amount; 

123 private S$unit; 

124 private $commodityType; 

123 

126 static S$totalRevenue = 0; 

2 static $numberOfTypes = 0; 

128 

129 function info(){ 

130 SrevenuePerType = S$this -> unitPrice * $this -> amount; 

131 SrevenuePortion = round ($revenuePerType / 

Revenue: :StotalRevenue * 100); 

132 return Sthis => anmounts™ "Sthis => unit.” "Sthis => 
commodityType. 

E33 " have been sold. The revenue is $".$revenuePerType.", ". 

134 "which takes ".$revenuePortion."%® of the total revenue"; 

135 ¢. 

136 

仙 3 流 function _construct($commodity, $price, $unit, $amount){ 

138 Sthis -> commodityType = $commodity; 

139 Sthis -> unitPrice = $price; 

140 Sthis -> unit = $unit; 

141 $this -> amount = $amount; 

142 

143 Revenue::$totalRevenue = Revenue: :StotalRevenue + S$this -> 
unitPrice * $this -> amount; 

144 Revenue::$numberOfTypes = Revenue: :SnumberOfTypes + 1; 

145 } 

146 

147 function _destruct(){ 

148 Revenue::$totalRevenue = Revenue: :StotalRevenue - S$this -> 
unitPrice * $this -> amount; 

149 Revenue: :$numberOfTypes = Revenue: :SnumberOfTypes - 1; 

150 } 

LS 


然后 ， 新 建 Ex8-12.php 文件 并 在 这 个 文件 中 输入 如 下 脚本 : 


p <?php 

多 require once "class.inc.php"; 

3 

4 $dailyRecords = array (new Revenue ("apples", 3.56, "kg", 320), 
5 new Revenue ("pears", 4.25, "kg", 400), 

6 new Revenue ("vaccume cleaners", 600, "", 25), 
洽 new Revenue ("TV sets", 800, "", 100), 

8 new Revenue ("basketball", 60, "", 127)); 

4 

10 echo "<ul>"; 

hl foreach ($dailyRecords as $dailyRecord){ 

2 echo "<]li>".$dailyRecord -> info()."</1Li>"7 

13 } 

14 echo "</ul>"; 

15 

16 Stotal = Revenue::$totalRevenue; 

和 $count = Revenue::$numberOfTypes; 

18 echo "Total Revenue: $".$total."<br>"; 

19 echo "Commodity Types: ".$count; 

20 ?2> 


在 Ex8-12.php 文件 里 的 这 段 脚本 中 ， 定 义 了 $dailyRecords 这 个 变量 ,然后 把 由 五 个 基 
于 Revenue 类 的 对 象 组 成 数组 赋值 给 它 。 接 着 使 用 了 foreach 语句 输出 了 关于 这 几 条 交易 记 
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录 组 成 的 报表 。 最 后 输出 了 总 收入 和 产品 类 型 总 数 。 
这 段 脚本 输出 的 结果 如 图 8-11 所 示 。 
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» 320 kg apples have been sold. The revenue is $1139.2, which takes 1% of the total revenue 
| 。 400 kg pears have been sold. The revenue is $1700, which takes 2% of the total revenue 
» 25 vaccume cleaners have been sold. The revenue is $15000, which takes 14% of the total 
revenue 
。 100 TV sets have been sold. The revenue is $80000, which takes 76% of the total revenue 
。 127 basketball have been sold. The revenue is $7620, which takes 7% of the total revenue 


Total Revenue: $105459.2 
Commodity Types: 5 


图 8-11 静态 属性 


如 果 仅 需要 对 一 个 类 的 静态 属性 进行 操作 ， 那 么 可 以 不 定义 任何 基于 该 类 的 对 象 直接 
修改 静态 属性 的 值 。 因 为 对 访问 静态 属性 使 用 的 是 “类 名 +::” 而 不 是 “Sthis->”。 比 如 我 
们 可 以 使 用 如 下 的 脚本 对 Revenue 类 中 的 totalRevenue 和 numberOfTypes 的 值 进行 修改 。 


1 require once "class.inc.php"; 


3 ”// 修 改 静 态 属性 totalRevenue 和 numberofTypes 的 值 
4 Revenue: :totalRevenue = 124000; 
5 Revenue::numberofTypes = 20; 


静态 方法 和 静态 属性 是 一 个 道理 , 都 是 在 类 这 个 层面 的 全 局 发 挥 作用 。 需要 注意 的 是 ， 
静态 方法 只 能 访问 静态 属性 , 而 不 能 访问 基于 该 类 的 对 象 , 因而 也 就 不 能 使 用 $this 关键 字 。 
我 们 也 可 以 在 不 定义 任何 基于 某 类 的 对 象 而 直接 访问 该 类 中 定义 的 静态 方法 。 假 如 ， 在 例 
8.12 中 的 Revenue 类 里 加 上 如 下 脚本 : 


static function total() { 
return Revenue::totalRevenue; 


} 


static function counts() { 
return Revenue::numberOfTypes; 


| 
然后 我 们 可 以 使 用 如 下 的 脚本 访问 这 两 个 静态 方法 : 


require once "class.inc.php"; 
// 显 示 静 态 属性 totalRevenue 和 numberOfTypes 的 值 


echo Revenue::totalRevenue; 
echo Revenue::numberofTypes; 


PODP 


83 对 象 


8.3.1 创建 对 象 


在 8.2 节 中 ， 我 们 创建 了 若干 个 类 ， 也 基于 这 些 类 创建 了 若干 个 对 象 。 在 本 小 节 里 
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我 们 就 来 总 结 一 下 如 何 创建 对 象 。 
在 例 8.7 中 ， 我 们 定义 了 一 个 Car 类 ， 然 后 在 Ex8-07.php 中 创建 了 一 个 基于 Car 类 的 
对 象 $fordFiesta。 在 创建 这 个 对 象 的 时 候 ， 使 用 了 new 关键 字 。 具 体 的 做 法 是 : 


$fordFiesta = new Car; 


在 这 一 句 脚本 中 ， 我 们 就 像 定 义 一 个 变量 一 样 来 定义 对 象 。 在 定义 这 个 对 象 时 : 
口 使 用 了 变量 定义 符 ($) 来 为 对 象 指定 对 象 名 (SfordFiesta); 
口 使 用 了 变量 赋值 符 (=) 来 指定 该 对 象 所 属 类 (Car) 。 
值得 注意 的 是 ， 类 名 前 是 不 使 用 变量 定义 符 的 。 
在 创建 了 对 象 之 后 ， 我 们 就 可 以 通过 这 个 对 象 使 用 该 对 象 所 属 类 的 各 种 属性 和 方法 
了 。 至 于 如 何 使 用 这 些 属 性 和 方法 ， 你 可 以 再 回 过 头 去 看 看 前 面 学 过 的 内 容 。 


8.3.2 ”克隆 对 象 


在 定义 一 个 类 的 时 候 ， 也 可 以 使 用 魔术 方法 _clone() 来 定义 当 基 于 该 类 的 某 个 对 象 被 
克隆 时 ， 该 对 象 的 属性 和 方法 应 该 如 何 运 作 。 如 果 在 定义 类 时 ， 没 有 使 用 _clone0 这 个 魔 
术 方 法 ， 则 当 基 于 该 类 的 某 个 对 象 被 克隆 时 ， 我 们 需要 使 clone 这 个 关键 字 ， 而 这 个 对 象 
的 所 有 属性 和 方法 在 被 克隆 的 一 刹那 所 处 的 状态 都 会 原封 不 动 的 被 克隆 。 

【 例 8.13】 克隆 对 象 的 使 用 。 

先 在 class.inc.php 中 写 入 如 下 脚本 : 


55 class GasFillerl { 

156 Private $gas = 100; 

57 private $operator = "SINOPEC"; 

158 

159 function sellGas($soldGas) { 

160 Sthis -> gas = $this -> gas - $soldGas; 

161 return Sthis -> gas." liters of gas are left. [".$this -> 
operator."]"; 

162 } 

163 

164 function _clone() { 

165 Sthis -> operator = "China Petrol"; 

166 } 

167 i 

168 

169 class GasFiller2 { 

PB private $gas = 100; 

TE private $operator = "SINOPEC"; 

E72 

hy function sellGas($soldGas) { 

174 Sthis -> gas = $this -> gas - $soldGas; 

人 return Sthis -> gas." liters of gas are left. [" .Sthis -> 

operator."]"; 

176 i 

E77 } 


这 段 脚本 定义 了 两 个 类 GasFillerl 和 GasFiller2 用 来 代表 两 台 加 油 机 。 第 一 台 加 油 机 
GasFillerl 定义 了 两 个 私有 属性 $gas 和 S$operator 来 代表 加 油 机 中 现存 油 量 和 运营 商 。 然 后 
定义 了 sellGas0 方 法 用 于 售 油 。 接 着 使 用 了 魔术 方法 _clone0 来 实现 当 基 于 该 类 的 对 象 被 
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克隆 时 修改 加 油 机 中 的 现在 油 量 和 运营 商 。 
在 第 二 台 加 油 机 GasFiller2 中 ， 除 了 没有 定义 _clone0 方 法 之 外 ， 其 他 的 脚本 与 
GasFillerl 一 样 。 接 着 ， 我 们 新 建 Ex8-12.php 文件 ， 然 后 在 该 文件 中 写 入 如 下 脚本 : 


1 <?php 
区 require once "class.inc.php"; 


3 

4 S$gasFillerl = new GasFillerl; 

5 echo S$gasFillerl -> sellGas(20); 
6 echo "<br>"; 

际 S$gasFiller2 = clone S$gasFillerl; 
8 echo S$gasFiller2 -> sellGas(20); 


echo "<br>"; 

10 SgasFiller3 = new GasFiller2; 
pl echo S$gasFiller3 -> sellGas(20); 
12 echo "<br>"; 

3 SgasFiller4 = clone S$gasFiller3; 
14 echo S$gasFiller4 -> sellGas(20); 
15 2> 


在 这 段 脚本 中 , 我 们 定义 了 四 台 加 油 机 , 它们 分 别 为 SgasFillerl、S$gasFiller2、S$gasFiller3 
和 S$gasFiller4。 其 中 ， 加 油 机 $gasFiller2 和 $gasFiller4 分 别 是 $gasFillerl 和 $gasFiller3 的 克 
隆 。 我 们 先 来 看 看 这 段 脚本 运行 的 结果 如 图 8-12 所 示 。 
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80 liters of gas are left. [SINOPEC] 
60 liters of gas are left. [China Petrol] 
80 liters of gas are left. [SINOPEC] 
60 liters of gas are left. [SINOPEC] 


图 8-12 克隆 对 象 


我 们 知道 两 个 加 油 机 类 中 的 Sgas 属性 的 初始 值 都 是 100。 通 过 图 8-12 所 示 的 结果 可 以 
知道 ， 加 油 机 $gasFiller2 只 克隆 了 $gasFillerl 中 的 现在 油 量 ， 而 重新 定义 了 运营 商 ， 而 加 
油 机 $gasFiller4 却 克 降 了 $gasFiller3 的 现存 油 量 和 运营 商 。 


8.3.3 销毁 对 象 


销毁 一 个 对 象 ， 和 销毁 一 个 变量 一 样 ， 都 要 使 用 unset0 方 法 。 在 8.2.2 小 节 中 ， 我 们 
提 到 了 魔术 方法 _destruct)。 如 果 我 们 在 定义 某 个 类 时 定义 了 _destruct() 方 法 ， 那 么 当 使 
用 unset() 方 法 销毁 一 个 基于 该 类 的 对 象 时 ，__destruct() 方 法 中 定义 的 脚本 会 被 执行 。 

【 例 8.14】 销毁 对 象 。 

首先 ， 在 class.inc php 文件 中 添加 一 个 Calc 类 。 脚 本 如 下 : 


ES class Calc { 

182 private $a; 
183 private $b; 
184 

185 function add(){ 
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186 return Sthis -> a + Sthis -> b; 
187 } 

188 

189 function subtract(){ 

190 return S$this -> a - $this -> b; 
191 i 

192 

193 function times(){ 

194 return Sthis -> a * Sthis -> bz 
卫 95 } 

196 

197 function divide(){ 

198 return Sthis -> a / $this -> b; 
二 99 } 

200 

201 function _construct($a, $b){ 

202 Sthis -> a = $a; 

203 Sthis -> b = $b; 

204 } 

205 

206 function destruct (){ 

207 echo "the Calc is switched off..."; 
208 } 

209 } 


这 个 类 一 共有 两 个 属性 〈$a 和 $b) 和 四 个 方法 〈 加 、 减 、 乘 、 除 ) 。 另 外 ， 还 定义 了 
两 个 魔术 方法 _construct0 和 destructO0， 前 者 用 来 赋值 ， 后 者 用 来 提示 基于 该 类 的 对 象 已 
销毁 。 
接 下 来 ， 新 建 Ex8-14.php， 然 后 在 文件 中 输入 以 下 脚本 : 
<?php 


require once "class.inc.php"; 


王 
2 
3 
4 SmyCalc = new Calc(5，10) 

5 echo $myCalc -> add()."<br>"; 

6 echo $myCalc -> subtract()."<br>"; 
yf echo $myCalc -> times()."<br>"; 

8 unset ($myCalc); 

9 echo SmyCalc -> divide()."<br>"; 
DON 


这 段 脚本 定义 了 基于 Calc 类 的 对 象 SmyCalc， 并 在 定义 gmyCalc 对 象 时 为 属性 Sa 和 $b 
赋 了 值 。 然 后 我 们 分 别 输出 了 两 个 属性 相 加 、 相 减 和 相 乘 的 值 ， 随 后 使 用 unset0 方 法 销毁 
了 $myCalc 类 。 最 后 试图 输出 属性 $a 除 以 属性 $b 的 值 。 这 段 脚 本 的 运行 结果 如 图 8-13 所 示 。 
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the Calc is switched off... 
Notice: Undefined variable: myCalc in C:\greatwall\chapter08\Ex8-14.php on line 9 


Fatal error: Call to a member function divide() on a non-object in C:\greatwall\chapter08 
\Ex8-14.php on line 9 


图 8-13 ”销毁 对 象 
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通过 图 8-13 可 知 ， 当 $myCalc 对 象 被 销毁 时 ， 系 统 输 出 了 如 下 的 提示 信息 : 


the Calc is switched off.. 
在 SmyCalc 对 象 被 销毁 之 后 ， 再 试图 使 用 该 对 象 时 ， 系 统 会 提示 我 们 SmyCalc 既 不 是 
对 象 ， 也 不 是 变量 。 


8.4 实战 练习 : 记 账 工具 (上 ) 


在 接 下 来 的 三 个 章节 的 实战 练习 中 ， 我 们 将 会 动手 制作 一 个 记 账 工 具 。 其 中 会 用 到 在 
这 三 章 中 学 习 到 的 知识 。 
本 节 先 来 写 一 个 记 账 工具 类 ， 为 其 定义 一 些 属 性 和 方法 。 
作为 账 务 系统 本 身 ， 它 的 基本 功能 应 该 有 : 单 笔 账 务 录入 、 批 量 账 务 录入 、 账 务 定期 
分 类 统计 和 账 务 格 式 化 。 同 时 ， 它 也 会 有 如 下 的 属性 : 记 账 使 用 的 货币 单位 和 统计 周期 等 。 
对 于 每 笔 账 务 而 言 ， 它 应 该 拥有 以 下 属性 : 账 务 描述 、 账 务 产 生 时 间 、 上 账 务 涉及 的 金 
额 和 账 务 所 属 分 类 。 
基于 上 述 分 析 ， 我 们 可 以 建立 一 个 bookKeeping 类 ， 用 来 涵盖 这 些 功 能 和 属性 。 
<?php 
/** 
* 创建 一 个 名 为 bookKeeping 的 类 
ee bookKeeping { 


// 初始 化 货币 单位 和 统计 周期 
function _construct() { 
Sthis -> unit = 'YUAN'; 
Sthis -> cycle = 'MONTH'; 
} 


// 检查 并 添加 一 条 消费 记录 
function AddSingleExpense ($desc, $amount, $category, $userId){ 
if(mb strlen($desc, 'UTF-8') <= 70 && 
intval ($amount) > 0 &E& 
strlen($userId) > 0) { 
$entryID = addExpenseToDatabase ($desc, $amout, $category, 
SuserId) 
return ' 添 加 消费 记录 ' .$entryID.' 成 功 .'; 
} else { 
return "参数 不 正确 。 添 加 消费 记录 失败 ' ; 


// 向 数据 库 中 添加 一 条 消费 记录 
private function addExpenseToDatabase ($desc, $amount, $category, 


$userId){ 


} 


// 检查 并 添加 一 条 收入 记录 


是 污 


第 3 篇 PHP 编程 基础 


function AddSingleIncome ($desc, $amount, $category, $userId){ 
if(mb strlen($desc, 'UTF-8') <= 70 && 
intval ($amount) > 0 &E& 
strlen($userId) > 0) { 
$entryID = addIncomeToDatabase ($desc, $amout, $category, 


$userId); 

return ' 添 加 收入 记录 ' .$entryID.' 成 功 .'; 
} else { 

return ' 参 数 不 正 确 。 添 加 消费 记录 失败 '; 


} 


// 向 数据 库 中 添加 一 条 收入 记录 
private function addIncomeToDatabase ($desc, $amount, $category, 
SuserId) { 


} 


> 

上 面 的 脚本 定义 了 一 个 名 为 bookKeeping 的 类 , 它 有 两 个 属性 , 分 别 为 “单位 Cunit) ” 
和 “ 记 账 周期 ”。 除 此 之 外 ， 它 还 有 若干 方法 分 别 用 来 添加 消费 和 收入 记录 。 值 得 注意 的 
是 ， 在 类 中 ， 还 定义 了 两 个 私有 的 方法 ， 用 于 向 数据 库 中 添加 记录 。 我 们 在 实例 化 一 个 类 
之 后 ， 不 能 直接 调用 类 的 私有 属性 和 方法 ， 而 只 能 通过 类 的 公有 方法 来 调用 。 在 这 个 例子 
里 , 通过 AddSingleIncome() 和 AddSingleExpense() 方 法 分 别 调用 了 AddIncomeToDatabase() 
和 AddExpenseToDatabase() 这 两 个 私有 方法 。 

另外 ， 因 为 这 两 个 私有 方法 涉及 到 数据 库 的 操作 ， 我 们 将 其 留 到 第 10 章 的 实战 练习 
中 进行 讲解 。 


8.5 习 题 


(1) 请 定义 一 个 可 以 返回 斜体 文字 的 函数 。 

(2) 请 定义 一 个 用 于 判断 指定 的 数字 是 否 为 100 以 内 的 偶数 的 函数 。 

(3) 请 创建 一 个 名 为 styler 的 类 ， 然 后 在 其 中 定义 bold0、italicize() 方 法 。 

(4) 请 创建 一 个 继承 自 styler 类 的 子 类 ， 并 将 其 命名 为 stylerEnhanced， 然 后 在 其 中 添 
加 underline() 方 法 。 

请 创建 一 个 水 果 分 捡 机 (fruitPicker) ， 用 来 分 捡 管 (basket) 中 的 水 果 。 黎 中 的 水 果 如 下 : 

apple,apple, apple, pear, pear, banana, apple, banana, pear, grapefruit, pear,ap 

ple,banana, banana, banana 

(5) 创建 一 个 名 为 fruitPicker 的 类 ， 该 类 带 有 一 个 字符 串 类 型 的 参数 Sbasket: 该 类 有 
两 个 私有 属性 ($fruitName 和 $fruitQuantity) 和 一 个 静态 属性 〈$totalQuantity) 。 

(6) 创建 一 个 名 为 applePicker 的 类 ， 该 类 继承 自 fruitPicker 类 。 定义 该 类 的 _construct 
魔术 方法 ， 使 得 该 类 被 实例 化 的 同时 ， 其 私有 属性 $fruitName 就 被 定义 为 "apple"。 再 按照 
类 似 的 步骤 分 别 创建 pearPicker 和 bananaPicker 类 。 

(7) 分 别 输出 苹果 、 梨 和 香 芍 的 数量 ， 以 及 篮 中 所 有 水 果 的 数量 。 
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在 之 前 的 8 章 里 ， 我 们 学 习 了 如 何 搭建 PHP 开发 环境 、 如 何 编写 PHP 脚本 、 如 何 使 
用 条 件 和 循环 语句 控制 脚本 的 运行 顺序 以 及 如 何 使 用 函数 、 类 和 对 象 来 复 用 脚本 。 在 掌握 
了 这 些 基础 知识 后 ， 终 于 要 开始 接触 Web 编程 了 。 

所 谓 的 Web 编程 ， 其 实 就 是 通过 编程 实现 基于 Web 浏览 器 的 人 机 互动 。 这 种 人 机 互 
动 要 求人 也 就 是 我 们 通常 说 的 用 户 ) 输 入 一 些 信 息 ， 然 后 机 (可 以 是 与 用 户 直 接 接触 的 
计算 机 、 该 计算 机 上 安装 的 浏览 器 , 或 者 是 远程 的 PHP 服务 器 ) 根据 用 户 输入 的 信息 做 出 
相应 的 反应 。 在 这 个 过 程 里 , 信息 的 输入 是 通过 HTML 实现 的 ， 输 入 信息 的 处 理 则 需要 使 
用 客户 端 脚 本 (如 JavaScript) ， 或 服务 器 端 脚本 (如 PHP) 来 实现 。 

那么 这 个 人 机 互动 的 过 程 是 如 何 实现 的 呢 ? 我 们 可 以 倒 回去 看 看 图 1-2。 在 那 幅 图 中 
可 以 知道 用 户 是 通过 浏览 器 向 服务 器 发 出 请 求 的 ， 这 些 请 求 里 可 能 会 带 有 某 些 用 户 输入 的 
信息 。 当 服务 器 收 到 这 些 请 求 之 后 , 会 将 其 中 用 户 输 入 的 信息 代入 到 PHP 脚本 中 进行 处 理 ， 
然后 将 处 理 的 结果 以 HTML 文件 的 形式 返回 到 浏览 器 ， 从 而 实现 人 机 互动 。 

用 户 在 通过 浏览 器 向 服务 器 发 出 请 求 和 传递 信息 时 ， 使 用 了 统一 资源 定位 符 (URL) 
来 确定 访问 对 象 并 向 访问 对 象 传递 数据 。 而 当 服 务 器 返回 信息 后 会 将 某 些 经 常会 使 用 到 的 
信息 以 cookies 的 形式 存储 在 本 地 缓存 中 以 加 快 信息 处 理 速 度 。 为 了 保障 人 机 互动 整个 过 
程 的 安全 ， 我 们 还 使 用 了 会 话 〈session) 机 制 来 避免 这 个 过 程 受到 恶意 攻击 。 有 具体 来 说 ， 
每 个 用 户 对 任意 一 个 访问 对 象 的 每 一 次 访问 都 可 以 被 看 做 是 一 个 会 话 。 

在 本 章 里 ， 我 们 就 来 具体 了 解 一 下 URL、cookies 和 sessions。 


9.1 使 用 URL 传递 数据 


几乎 所 有 的 支持 人 机 交互 的 网 站 都 是 使 用 URL 和 表单 来 传递 数据 的 。 现 在 以 某 搜索 
引擎 为 例 ， 来 看 看 如 何 使 用 URL 传递 数据 。 
首先 在 该 搜索 引擎 的 首页 上 输入 “PHP”, 然后 单 击 “ 搜 索 ” 按 钮 。 搜 索 结果 如 图 9-1 所 示 。 


图 9-1 使 用 搜索 引擎 搜索 PHP 相关 网 页 
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在 图 9-1 所 示 中 ， 重 点 关注 一 下 搜索 结果 页 面 的 URL (也 就 是 我 们 常 说 的 网 址 ) 是 个 


十 


http://www.searchengine.com/s?ie=utf-8&src=360sou_home&q=PHP&_re=0 


一 和 
0 @ 5 


图 9-2 搜索 结果 页 面 的 URL 


在 上 面 这 个 网 址 中 ,我 们 可 以 看 到 有 6 个 部 分 。 在 这 6 个 部 分 中 ， 除了“?” 前 面 的 文 
本 传输 协议 名 称 、 域 名 及 文件 名 外 ，“?” 后 面 的 四 个 部 分 就 是 通过 URL 传递 的 参数 字段 : 

口 字段 1 的 参数 名 为 ie， 参 数值 为 utf-8， 用 来 表示 浏览 器 和 页 面 编码 类 型 。 

口 字段 2 的 参数 名 为 src， 参 数值 为 360sou home， 用 来 表示 搜索 来 源 页 面 。 

口 字段 3 的 参数 名 为 g， 参 数值 为 PHP， 用 来 表示 搜索 关键 字 。 

口 字段 4 的 参数 名 为 re， 参 数值 为 0， 具体 代表 的 意义 不 其 了 了 。 

值得 注意 的 是 ， 所 有 的 参数 字段 都 放 在 “?” 后 并 使 用 “&” 符 号 连接 。 

我 们 只 是 在 搜索 框 里 输入 了 一 个 简单 的 关键 字 ， 结 果 却 产生 了 四 个 关联 的 参数 字段 。 
在 这 四 个 参数 字段 中 ， 字 段 1 应 该 是 通过 引用 某 系 统 常 量 得 出 的 ， 字 段 2 和 字段 4 应 该 是 
由 提交 隐藏 文本 框 传递 的 , 而 字段 3 就 是 我 们 在 搜索 页 面 上 输入 的 内 容 。 从 这 里 可 以 看 出 ， 
我 们 可 以 要 求 用 户 使 用 表单 输入 某 些 信息 ， 然 后 将 这 些 信息 通过 URL 传递 到 其 他 页 面 。 

在 这 一 节 里 ， 我 们 就 来 详细 了 解 一 下 如 何 实现 这 一 过 程 。 


9.1.1 收集 用 户 信息 


如 果 大 家 对 HIML 有 一 定 的 了 解 ， 应 该 知道 表单 是 什么 。 例 9.1 就 展示 了 一 段 HIML 
代码 ， 我 们 一 起 来 看 看 。 

【 例 9.1】 收集 信息 。 

先 新 建 一 个 Ex9-01.html 文件 ， 然 后 在 文件 中 输入 如 下 代码 。 


二 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/ 
TR/html4/strict.dtd"> 


当 <meta content="text/html; charset=UTF-8" /> 

3 <html> 

4 <head> 

5 <title>A Form</title> 

6 </head> 

<body> 

8 <h2>Enter your name, please!</h2> 

全 <form action="Ex9-01.php" method="get"> 
10 <h3>Name</h3> 

a <input type="text" name="fullname" /> 
2 <input type="submit" value="Submit" /> 
13 </form> 
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14 </body> 

15 </html> 

在 这 段 HTML 代码 里 ， 我 们 重点 关注 一 下 form 标签 里 的 action 属性 和 method 属性 。 
action 属性 的 值 是 一 个 文件 名 ， 也 就 是 说 通过 这 个 表单 提交 的 数据 会 被 传递 到 Ex9-01.php 
里 进行 处 理 ， 而 method 属性 的 值 是 get， 也 就 是 说 这 些 通过 这 个 表单 提交 的 数据 会 以 明文 
方式 传递 到 通过 action 属性 指定 的 页 面 中 。 

现在 ， 新 建 一 个 文件 ， 将 其 命名 为 Ex9-01.php。 接 着 ， 运 行 Ex9-01.html、 在 文本 框 中 
输入 名 字 、 单 击 “ 提 交 ” 按 钮 。 发 现 Ex9-01.html 消失 了 ， 取 而 代 之 的 则 是 Ex9-01.php， 如 
图 9-3 所 示 。 


herp /eww greatwall tea chapeer00 -0 html 


会 Fovertes 由 AFom 站 

= _» 
Enter your name, please! 

Name 


Ga [Er 


re 
CHA hapywwwgreewattteaychaptensyE6.0Lphpyfulname=Good-job 昌国 四 四 三 


| 突 Favoritef | @ http://www.greatwalltea/chapter09/E9-01.php?.. 从 "车 


http://www.greatwall.tea/chaptere9/Ex9-81.php?fullname=Good+Job 


图 9-3 提交 表单 之 后 


但 是 ， 页 面 的 网 址 后 半 部 分 居然 带 有 参数 字段 。 仔 细 打量 一 番 ， 这 个 字段 的 参数 名 是 
“fullname”， 而 参数 值 则 是 “Good+Job”: 前 者 是 表单 中 文本 框 元 素 的 name 属性 值 ， 而 
后 者 则 是 在 Ex9-01.html 中 输入 的 名 字 。 

这 样 一 来 ， 数 据 就 由 Ex9-01.html 传递 到 了 Ex9-01.php 中 了 。 

需要 说 明 的 是 ， 使 用 明文 传递 数据 会 比较 危险 ， 很 容易 造成 信息 泄露 。 为 了 防止 出 现 
这 种 情况 ， 我 们 可 以 使 用 post 方法 来 隐藏 被 传递 的 字段 。 不 过 这 样 一 来 ， 服 务 器 的 开销 无 
形 中 就 变 大 了 ， 一 旦 访问 流量 过 大 ， 就 很 容易 出 现 资源 耗 尽 、 服 务 器 宕 机 的 情况 。 其 实 我 
们 可 以 使 用 其 他 的 方式 来 对 以 明文 方式 传递 的 数据 进行 加 密 ， 从 而 在 降低 服务 器 开销 的 同 
时 ， 也 保护 了 数据 的 私密 性 。 


9.1.2 ”接收 信息 数据 
上 一 小 节 里 ， 我 们 使 用 表单 向 一 个 页 面 传递 了 从 用 户 处 收集 到 的 信息 。 在 本 小 节 里 


就 来 看 看 如 何 接收 这 些 数据 。 
【 例 9.2】 接收 数据 的 使 用 。 


在 Ex9-01.php 中 输入 如 下 代码 : 
1 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN" 
涩 "http:/V/www-w3-org/TR/htm14/strict-dtd"> 


i 
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3 <meta content=t"text/html; charset=UTF-8" http-equiv="Content-Type" /> 
4 <html> 

5 <head> 

6 <title>Welcome!'</title> 

ya </head> 

8 <body> 

3 <h2> 

10 Welcome! <?php echo $ GET['fullname'];?> 
11 </h2> 

12 </body> 

13 </html> 


在 这 段 代 码 中 , 使 用 了 一 段 PHP 脚本 用 来 输出 从 Ex9-01.html 的 表单 中 收集 到 的 数据 。 
现在 我 们 再 次 打开 Ex9-01.html 文件 ， 输 入 一 个 名 字 ， 看 看 结果 到 底 如 何 ? 


eptet ES hm 
ar 

Enter your name, please! 

Name 


CT] IE 


Se。 pwww greetwattea'chapteos/Ee-olLphpyuiname=Good-ob 


窒 Favortes | 逢 cite>Wekcome 偷 ” 园 - 己 ， 


Welcome! Good Job 


图 9-4 接收 数据 


按照 图 9-4 所 显示 的 结果 ， 在 Ex9-01.html 中 输入 的 名 字 成 功 地 显示 在 了 Ex9-01.php 
中 。 大 家 可 以 重点 关注 一 下 例 9.2 中 加 粗 的 那 一 段 代码 。 在 这 段 代 码 中 使 用 了 $_GET 数组 ， 
用 来 获取 表单 中 指定 元 素 的 值 。 

PHP 一 共 为 我 们 提供 了 三 个 预 置 数 组 ， 用 来 存储 从 表单 中 获取 的 数据 。 除 了 刚才 使 用 
的 $_GET 数组 之 外 ， 还 有 $_POST 数组 和 $_REQUEST 数组 。$_GET 数组 只 有 当 表 单传 递 
数据 的 method 设置 为 get 时 才 会 生效 。 类 似 地 ,$_ POST 数组 只 有 当 表单 传递 数据 的 method 
设置 为 post 时 才 会 生效 。 而 $_REQUEST 数组 则 对 任意 一 种 数据 传递 方式 都 有 效 。 

为 了 加 深 印 象 ， 再 来 做 一 个 比较 复杂 一 点 的 。 在 例 9.3 所 示 中 ，Ex9-02.html 里 有 四 个 
参数 需要 传递 到 Ex9-02.php 中 进行 处 理 ,它们 分 别 是 : 姓名 、 性 别 、 出 生日 期 和 婚姻 状况 。 
Ex9-02.php 页 面 在 接收 到 Ex9-02.html 传递 过 来 的 数据 后 , 根据 性 别 、 出 生日 期 和 婚姻 状态 
的 不 同 ， 显 示 不 同 的 问候 语 。 

【 例 9.3】 属于 你 的 问候 。 

新 建 Ex9-02.html 文件 ， 在 其 中 输入 如 下 HTML 代码 : 


于 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" 

2 "http://www.w3.o0rg/TR/html4/loose.dtd"> 

可 <meta content="text/html; charset=UTF-8" http-equiv="Content-Type"> 
4 <html> 

5 <title> 

6 Please fill the form! 


"180。 
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45 
46 


</title> 


<body> 


<h2> 


Please fill the form! 


</h2> 
<form action="Ex9-02.php" method="get"> 


<p> 
First Namegnbsp; 
<input type="text" name="firstName" /> 
</p> 
<p> 
Last Namegnbsp; 
<input type="text" name="lastName" /> 
</p> 
<p> 
Gendergnbsp; 
<input type="radio" name="urGender" value="Male" 
checked="" /> 


Male 
<input type="radio" name="urGender" value ="Female" /> 
Female 
</p> 
<p> 
Birthdayg&nbsp; 
<input type="text" name="urBirthday" value="1980-01- 
bh Bd > 
&nbsp; * Eg:1980-01-01 
</p> 
<p> 


Marital Statusgnbsp; 
<select name="urMarriage"> 
<option value="Married">Married</option> 
<option value="Unmarried">Unmarried</option> 
</select> 
</p> 
<p> 
<input type="submit" value="Submit" /> 
<input type="reset" value="Reset"/> 
</p> 


</form> 


</body> 
</html> 


接 下 来 ,再 新 建 Ex9-02.php 文件 ， 用 来 接收 和 处 理 Ex9-02.html 文件 中 传递 来 的 数据 。 
还 记得 我 们 应 该 如 何 接 收 由 其 他 文件 传递 来 的 数据 吗 ? 


<?php 


// 从 Ex9-02.html 中 获取 数据 
SfirstName = $ REQUEST['firstName']; 


$lastName = $ REQUEST['lastName']; 
S$gender = $ REQUEST['urGender']; 
$birthday = $ REQUEST['urBirthday']; 


Smarriage = $ REQUEST['urMarriage']; 


// 计 算 用 户 的 年 龄 

function getAge ($birthday) { 
//obtain the year in $birthday 
SbirthYear = substr($birthday, 0, 4); 


//obtain the current year 
$currentYear = date("Y", time()); 


-ls 
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43 
44 
45 


//calculate the age 
return $currentYear - S$birthYear; 
$ 
// 给 用 户 定 称谓 
function getTitle($gender, $marriage) { 
if (substr($gender, 0, 1) == "M") { 
return "Mr.™; 
} else { 
if (substr ($marriage, 0, 1) == "M") { 
return "Mrs。"s 
} else { 
return "Miss"; 
} 
} 
1 
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" 


"http://www.w3.org/TR/html4/loose.dtd"> 
<meta content="text/html; charset=UTF-8" http-equiv="Content-Type"> 
<html> 

<title>Welcome</title> 


<body> 
<h1l> 
Welcome, <?php echo getTitle($gender, $marriage)." 
".$ REQUEST['lastName'] ?>! 
</h1> 
<p>You are <?php echo getAge ($birthday) ?> years old now!</p> 
</body> 
</html> 


这 段 脚本 首先 定义 了 五 个 变量 用 来 获取 从 Ex9-02.html 中 传递 过 来 的 数据 ， 然 后 定义 
了 两 个 函数 getAge() 和 getTitle0) 来 确定 用 户 的 年 龄 和 称谓 。 现 在 运行 Ex9-02.html， 并 输入 
如 图 9-5 所 示 的 内 容 ， 然 后 单 击 “Submit” 按 钮 。 在 页 面 跳 转 到 Ex9-02.php 后 ， 我 们 在 
Ex9-02.html 中 的 表单 里 填写 的 所 有 内 容 都 可 以 在 Ex9-02.php 的 URL 中 找到 。 


“B22 


Prip// wwe greatwalltea chapeer03 /E02 Nm 


窜 Fovortes  @ Pleasefiltheform 


Please fill the form! 


First Name Julie 


Last Name Anderson 
Gender © Male ® Female 


Birthday 1974.02-1 相 


Marital Status [Maried [z=] 


Ra 


Welcome, Mrs. Anderson! 


You are 39 years old now! 


http://www.greatwall.tea/chapter09/ 
Ex9-02.php?firstName=Julie&lastName=Anderson&urGender=Female&urBirthday=1974-02-18&urMarriage=Married 


图 9-5 属于 你 的 问候 
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现在 ， 试 着 修改 一 下 Ex9-02.html 表单 里 各 项 目的 值 ， 看 看 结果 有 什么 变化 吧 ? 
9.1.3 检测 接收 到 的 数据 


假如 在 Ex9-02.html 中 ， 输 入 了 如 图 9-6 所 示 的 内 容 。 那 么 Ex9-02.php 中 显示 的 问候 
可 能 会 有 些 不 妥 。 


,http /wow greatwalltealchepte09/ES-02 html 


傅 hvontes | 乱 Pleasefilltheform! 


Please fill the form! ey 
First Name 8793 


Last Name 904534 


Gender ® Male © Female ~ 
Birthday Anderson| es CT Sr 
本 喇 


Marital Status [Maried [el 富 Fovortes | 乱 wacome me 


[erly 


Welcome, Mr. 904534! 


| You are 2013 years old now! 


图 9-6 不 妥 的 问候 


如 图 9-6 所 示 ， 我 们 在 First Name 和 Last Name 两 个 栏目 里 输入 了 数字 ， 而 在 应 该 填 
写 日 期 的 生日 一 栏 里 却 输入 了 人 名 。 这 样 一 来 ， 虽 然 系 统 没 有 报错 ， 但 是 出 现 的 结果 却 十 
分 的 怪异 。 

为 了 避免 出 现 这 种 乾 粹 的 情况 ， 我 们 需要 对 从 其 他 页 面 接收 到 的 数据 进行 检测 。 如 果 
接收 到 的 某 些 数据 不 符合 要 求 ， 就 应 该 提示 用 户 进行 必要 的 修改 。 

现在 就 着 手 对 Ex9-02.php 进行 改造 : 为 其 添加 数据 检测 的 功能 。 

【 例 9.4】 参数 检测 。 

首先 ， 将 Ex9-02.html 复制 一 份 ， 并 将 其 重 命名 为 Ex9-03.html。 然 后 将 文件 中 的 第 12 


行 由 : 
12 <form action="Ex9-02.php" method="get"> 
改 成 : 
人 <form action="Ex9-03.php" method="get"> 
接着 ， 新 建 Ex9-03.php 文件 ， 并 在 文件 中 输入 : 
. <?php 
加 // 获 取 从 Ex9-03.html 传递 来 的 数据 
3 $firstName = $ REQUEST['firstName']; 


= 3 
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4 $lastName = $ REQUEST['lastName']; 

3 S$Sgender = $ REQUEST['urGender']; 

6 $birthday = $ REQUEST['urBirthday']; 

Smarriage = $ REQUEST['urMarriage']; 

8 $errMsgHead = "Attention!"; 

9 $errMsgCont = array('03256' => "Invalid name. 

10 A name must start with a uppercase letter 

I and contain only letters.", 

2 "04218' => "Invalid date. 

3 The format of a valid birthday is similar 

to 

14 1980=01=12.")> 

15 

16 $errCount = 0; 

1 $errMsg = 

18 

19 /用 于 检测 $firstName 和 $lastName 是 否 正确 的 函数 

20 function validateName ($name) { 

亿 于 return preg match("/^[A-2Z] [a-z] *$/", $name); 

2 } 

2 

24 // 用 于 检测 $birthday 的 值 是 否 正确 函数 

25 function validateBirth($birthday) { 

26 return preg match("/^[0-9] {4} (\- (0[1-9] 11[0-2])) (\-(0[1-9]1 
1[0-9] 12[0-9]13[0-1]))$/", $birthday); 

加 } 

28 

29 // 开 始 检 测 相 关 参 数 

30 $is firstName = validateName ($firstName); 

eh $is lastName = validateName ($lastName); 

3 $is birthday = validateBirth ($birthday); 

33 

34 // 计 算 用 户 的 年 龄 

35 function getAge (Sbirthday) { 

36 // 获 取 $birthday 中 的 年 份 

37 $birthYear = substr ($birthday, 0, 4); 

38 

39 // 获 取 当 前 年 份 

40 ScurrentYear = date("Y", time()); 

41 

42 // 返 回 用 户 年 龄 

43 return S$currentYear - S$birthYear; 

44 } 

45 

46 // 获 取 用 户 称谓 

47 function getTitle($gender, S$marriage) { 

48 if (substr($gender, 0, 1) == "M") { 

49 return "Mr."; 

50 } else { 

if (substr($marriage, 0, 1) == "M") { 

S2 return "Mrs."; 

EE } else { 

54 return "Miss"7 

5 } 

56 |: 

yh ’ 

SO 

59 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" 

60 "http://www.w3.o0rg/TR/html4/loose.dtd"> 


61 <meta content="text/html; charset=UTF-8" http-equiv="Content-Type"> 
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62 <html> 

63 <head> 

64 <?php 

65 if ($is birthday == FALSE || 

66 $is firstName == FALSE || 

67 $is lastName == FALSE){ 

68 echo $errMsgHead; 

69 } else { 

70 echo "Welcome™"; 

ye i 

人 ?> 

73 </head> 

74 <body> 

75 <?php 

76 if ($is birthday == FALSE || 

Tn $is firstName == FALSE || 

78 $is lastName == FALSE){ 

19 2 

80 Before you continue, ratify the following errors: 

81 wl 

82 <?php 

83 if ($is firstName == FALSE || $is lastName == FALSE) 

84 echo "<]li>".$errMsgCont['03256"']."</l1i>"; 

85 if ($is birthday == FALSE) 

86 echo "<li>".S$errMsgCont['04218']."</1i>"; 

87 2> 

88 </ul> 

89 <?php 

90 } else { 

3 2> 

92 <hl>Welcome, <?php echo getTitle($gender, $marriage)." 
".$ REQUEST['lastName'] ?>!'</h1> 

3 <p>You are <?php echo getAge ($birthday) ?> years old now!</p> 

94 <?php 

95 

96 2> 

97 <div align="center"> 

98 <a href="Ex9-03.html">Return</a> 

99 </div> 

100 </body> 


101 </html> 


在 编写 完 这 两 个 文件 后 ， 运 行 Ex9-03.html， 然 后 在 表单 中 输入 如 图 9-7 所 示 的 内 容 ， 


单 击 “Submit” 按 钮 。 期 待 中 的 欢迎 页 面 并 未 出 现 ， 取 而 代 之 的 却 是 两 条 错误 提示 。 这 就 


意味 着 ， 我 们 在 Ex9-03.php 中 编写 的 参数 检测 机 制 发 挥 了 作用 。 

在 实现 参数 检测 机 制 的 过 程 中 ， 我 们 使 用 了 许多 之 前 学 过 的 内 容 ， 包 括 数组 、 正 则 表 
达 式 和 自 定义 函数 等 。 在 这 里 一 起 来 详细 地 讲解 这 段 代 码 。 

具体 来 说 ， 在 上 面 这 段 脚 本 里 完成 了 如 下 几 项 任务 。 


(1) 接收 F 


Ex9-03.html 传递 来 的 数据 


在 脚本 的 2 一 7 行使 用 $_REQUEST[] 获 取 了 由 Ex9-03.html 传递 来 的 数据 ,并 将 它们 赋 
值 给 若干 变量 备用 。 

(2) 定义 出 错 信息 

由 于 在 脚本 中 添加 了 参数 检测 机 制 ， 因 此 需要 在 用 户 输入 的 信息 不 符合 要 求 时 给 予 用 
户 相应 的 提示 。 在 脚本 的 8 一 14 行 定 义 了 一 个 变量 gerMsgHead 和 一 个 数组 SerrMsgCont 
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用 来 存放 提示 信息 。 

(3) 定义 用 于 检测 关键 参数 的 函数 ， 并 检查 关键 参数 

在 脚本 的 16 一 27 行 定义 了 validateName() 和 validateBirthO 两 个 函数 来 检查 用 户 输入 的 
姓名 和 出 生日 期 是 否 符合 格式 要 求 。 在 这 两 个 函数 中 ， 我 们 使 用 了 正则 表达 式 和 
preg_match() 函 数 。 然 后 在 29 一 32 行 里 ， 对 用 户 输入 的 First Name、Last Name 和 Birthday 
进行 检查 ， 并 将 检查 结果 放 入 $is_firstmame、Sis_lastname 和 $is_birthday 三 个 变量 中 。 

(4) 计算 用 户 年 龄 ， 获 取 用 户 称谓 

在 脚本 的 34 一 57 行 定义 了 getAge0 和 getTitle0 两 个 函数 来 根据 用 户 输入 的 出 生日 期 来 
计算 用 户 年 龄 ， 根 据 用 户 选择 的 性 别 和 婚姻 状况 来 确定 用 户 的 称谓 。 

(5) 确定 Ex9-03.php 页 面 的 标题 

在 脚本 的 64 一 72 行使 用 站 .else 语句 判断 了 Sis_firstname、S$is_lastname 和 $is_birthday 
三 个 变量 的 值 。 只 要 这 三 个 变量 中 的 值 任意 一 个 为 FALSE， 则 页 面 标题 为 SerMsgHead 的 
值 ， 和 否则 页 面 标题 为 “Welcome”。 

(6) 确定 Ex9-03.php 页 面 内 容 

在 脚本 的 74 一 100 行使 用 了 站 .else 语句 的 嵌 套 ， 以 便 根据 用 户 输入 的 内 容 动态 显示 
不 同 的 内 容 。 若 用 户 输入 的 First Name 或 Last Name 不 符合 格式 要 求 ， 则 Ex9-03.php 的 页 
面 标题 为 “Attention!”， 页 面 内 容 包 含 了 一 条 关于 “Invalid name.” 的 提示 。 若 用 户 输入 
的 Birthday 不 符合 格式 要 求 ，Ex9-03.php 的 页 面 标题 为 “Attention!”， 页 面 内 容 则 包含 了 
一 条 关于 “Invalid date.” 的 提示 。 若 这 三 个 关键 参数 的 格式 均 不 正确 ，Ex9-03 的 页 面 标题 
仍然 为 “Attention!”， 页 面 内 容 则 包含 了 上 述 两 条 错误 提示 。 只 有 当 这 三 个 关键 参数 的 格 
式 均 正确 时 ，Ex9-03 的 标题 和 内 容 才 会 如 图 9-5 所 示 。 


痪 Fwortes 镍 Plessefilltheform! 


Please fill the form! ee 
First Name 8793 


Last Name 904534 


Gender ® Male © Female 


Attentio 


es EY SE 


Birthday Andersor| 
Marital Status [Maried [= 


[Som | [Reset | 


宣 Favortes 。 忽 Attention! 价 ~- 国 - 己 


Before you continue, ratify the following errors- 


* Invalid name A name must start with a uppercase letter and contain only letters 
* Invalid date The format of a valid birthday is similar to 1980-01-12. 


Retun 


图 9-7 参数 检测 


总 结 一 下 ,在 上 面 的 这 段 脚 本 里 , 接收 了 由 其 他 页 面 传递 来 的 数据 、 检 查 数 据 的 格式 、 
根据 用 户 输入 的 数据 进行 用 户 年 龄 和 称谓 的 计算 ， 最 后 输出 计算 结果 。 若 用 户 输入 的 数据 
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不 符合 要 求 ， 输 出 提示 信息 帮助 用 户 改 正 。 
9.2 使 用 Cookie 缓存 数据 


大 家 上 网 的 时 间 一 长 ， 就 会 产生 很 多 的 隐私 数据 。 这 些 隐 私 数据 除了 占据 了 本 地 存储 
空间 之 外 , 还 有 可 能 造成 隐私 泄露 。 因 此 ,大 多 数 的 浏览 器 都 提供 了 清理 浏览 数据 的 功能 。 
而 Cookie 就 是 隐私 数据 的 一 种 。 图 9-8 展示 了 由 Google 开发 的 开源 浏览 器 Chrome 中 清理 
浏览 数据 的 界面 。 


入 议 旦 - 油 KWIS 扩 


€ 3 © |D chrome//cvome/setings/clearBrowserData 
Suggested See ) Web Sfice Gallery Imported FromIE 


污 际 浏 记 数 生 


这 和 则 Bs 的 DT 项 : | 这 二 二 小时 二 

Wi 

大 下 委 历 实 记 录 

Rs 

网 测 队 Cookie 以 及 其他 网 Ri 所 各 白人 条 所 
再 ER 二 汪 


了 汪汪 


图 9-8 ”Google Chrome 浏览 器 清除 浏览 数据 


其 中 ， 第 四 项 即 为 “删除 Cookie 以 及 其 他 网 站 数据 和 插件 数据 ”。 那 么 ， 到 底 什么 是 
Cookie 呢 ? 按照 维基 百科 的 解释 ，Cookie 是 某 些 网 站 为 了 辨识 用 户 身份 而 存储 在 本 地 终端 
上 的 经 过 加 密 的 数据 。 

这 说 明 , 通过 Cookie 缓存 的 数据 有 可 能 被 用 户 自行 清空 而 无 法 获取 。 因 此 ， 指 望 通过 
这 种 方法 将 数据 缓存 到 本 地 基本 上 是 不 太 现实 的 事情 。 即 使 如 此 ， 我 们 还 是 有 必要 了 解 一 
下 ， 如 何在 本 地 缓存 少量 数据 ， 以 达到 提升 用 户 浏览 体验 的 目的 。 


9.2.1 使 用 Cookie 存 取 数 据 


在 PHP 中 ， 可 以 使 用 setcookie 方法 在 用 户 本 地 存储 Cookie 信息 。 具 体 用 法 如 下 : 


setcookie ("variable", "value"); 


存储 在 用 户 本 地 的 Cookie 数据 与 通过 URL 传递 的 数据 的 格式 一 样 ， 都 是 键 值 对 。 也 
就 是 说 上 面 这 条 语句 中 的 variable 是 “ 键 ”， 而 value 则 是 “ 值 ”。 需 要 注意 的 是 ,在 “ 键 ” 
的 前 面 是 不 需要 使 用 “$” 符 号 的 。 例 如 ， 我 们 可 以 使 用 下 面 这 条 语句 ， 把 “prov=Beijing” 
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这 条 信息 存储 在 本 地 。 

Setcookie ("prov", "Beijing"); 

当 用 这 条 语句 把 信息 存储 在 本 地 后 , 可 以 使 用 PHP 内 置 的 数组 $ COOKIE 来 提取 这 些 
信息 。 比 如 ， 可 以 使 用 下 面 这 条 语句 来 显示 我 们 刚才 存储 在 用 户 本 地 的 信息 : 


echo "You are from ".$ COOKIE['prov']."."; 


这 条 语句 输出 的 结果 应 该 是 : 


You are from Beijing. 


需要 注意 的 是 , 在 设置 $_COOKIE 数组 的 页 面 中 ， 系统 无 法 读 取 Cookie 数据 。 只 有 当 
用 户 离开 该 页 面前 往 其 他 页 面 时 ， 才 可 以 使 用 5_COOKIE 数组 读 取 已 存储 的 Cookie 信息 。 

通常 情况 下 ， 我 们 会 把 用 户 名 等 一 些 涉 及 用 户 隐 私 的 信息 通过 Cookie 存储 在 用 户 本 
地 ， 从 而 方便 用 户 身份 的 认证 。 但 是 这 也 给 不 怀 好 意 的 人 有 机 可 乘 。 为 了 尽 可 能 的 避免 这 
种 情况 ， 可 以 给 Cookie 信息 设置 有 效 期 。 

setcookie ("variable", "value", expiretime); 

在 这 条 语句 中 ， 前 两 个 参数 与 之 前 无 异 ; 最 后 一 个 参数 expiretime 指 的 是 该 Cookie 过 
期 的 时 间 点 。 我 们 可 以 使 用 之 前 学 过 的 time0 或 mktime0 函 数 来 指定 。 

口 timeO 函 数 以 时 间 戳 的 形式 返回 当前 的 系统 时 间 ， 单 位 为 秒 可 以 在 其 后 加 上 一 定时 

间 的 秒 数 来 设 定 一 个 未 来 的 时 间 点 ， 比 如 : 

setcookie ("prov", "Beijing", time() + 3600); //Cookies 将 在 1 小 时 后 过 期 

setcookie ("Name"，S$name, time() + (3*86400)); //Cookies 将 在 3 天 后 过 期 

口 mktimeO 函 数 则 可 以 直接 指定 一 个 时 间 点 ， 比 如 : 

setcookie ("prov"，"Beijing"，mktime(3，0，0，4，1，2013) );//Cookies 将 在 

2013 年 4 月 1 日 时 上 3: 00 过 期 


需要 注意 的 是 mktimeO 函 数 的 参数 分 别 为 : 时 、 分 、 秒 、 月 、 日 、 年 。 


9.2.2 ”销毁 Cookie 数据 


在 为 一 条 Cookie 数据 设置 了 过 期 时 间 点 后 ， 时 间 一 到 ， 数据 就 会 自动 失效 , 我 们 也 就 
无 法 通过 $_COOKIE 数组 调 取 该 条 数据 了 。 当然, 如 果 你 想 在 该 条 数据 到 期 前 手动 销毁 它 ， 
可 以 使 用 setcookie0 方 法 重新 定义 该 Cookie 的 键 对 值 ， 将 其 值 设置 为 空 即 可 ， 比 如 : 


setcookie ("prov", ""); 


9.2.3 ”关于 Cookie 的 后 话 


在 知道 如 何 使 用 Cookie 存 取 数据 和 销毁 数据 后 。 还 是 想 跟 大 家 讲 讲 何 时 使 用 Cookie。 

由 于 Cookie 数据 存储 在 用 户 本 地 , 越 来 越 多 的 用 户 对 这 种 机 制 心 存 芥 蒂 。 许 多 的 用 户 
干脆 在 本 地 禁用 了 Cookie 机 制 , 从 而 禁止 所 有 的 网 站 于 用 户 浏览 时 在 本 地 存 取 数据 ,因此 ， 
不 要 过 度 的 依赖 Cookie 机 制 在 用 户 本 地 存储 大 量 的 信息 。 一 旦 用 户 禁 用 了 Cookie 功能 ， 
所 有 的 数据 以 及 与 这 些 数据 相关 的 功能 都 会 失效 。 
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另外 ， 还 需要 注意 的 是 ，setcookie0 方 法 有 其 局 限 性 。 不 能 在 PHP 脚本 已 经 向 页 面 输 
出 信息 后 再 定义 Cookie 数据 ， 和 否则 setcookie0 无 效 。 


9.3 使 用 Session 保障 数据 安全 


与 Cookie 数据 存储 在 用 户 本 地 不 同 ，Session 数据 存储 在 服务 器 上 。 一 个 Session 是 指 
某 用 户 在 某 网 站 上 逗留 的 整 段 时 间 ， 这 段 时 间 可 以 称 其 为 一 个 会 话 期 间 ， 或 简称 其 为 一 条 
会 话 。 在 用 户 开 始 进入 一 个 网 站 到 离开 该 网 站 的 这 段 时 间 内 ， 我 们 可 以 使 用 Session 机 制 
使 该 网 站 的 所 有 页 面 共享 某 些 数据 ， 从 而 提升 用 户 的 浏览 体验 。 


9.3.1 PHP Session 工作 机 制 


我 们 可 以 使 用 PHP 脚本 创建 和 存储 Session 变量 。 在 创建 了 一 个 Session 后 ,所 有 Session 
变量 在 用 户 一 次 会 话 期 间 里 访问 的 所 有 页 面 都 有 效 。 其 工作 机 制 如 图 9-9 所 示 。 


PHP 为 Session 分 配 Session ID 


] 


PHP 将 该 会 话 期 间 产生 的 所 有 


Session 安 量 仓储 人 php ini 文 件 中 
指定 的 Session 文件 目录 中 的 以 该 
Session ID 命名 的 文件 中 


PHP 通 过 Cookie 缓存 
Session ID ， 使 得 所 有 页 面 
都 可 以 获取 该 Session ID PHP 使 用 
$_SESSION 数 组 访 
间 Session ID 同名 
的 Session 文件 获 
取 所 需 的 Session 
变量 
检测 是 否 开启 PHP 通 过 SID 常 量 在 页 面 间 
了 trans -sid 功 能 传递 Session ID 


Session 只 有 通过 手动 的 方式 才 
可 以 在 页 面 间 传 递 


图 9-9 PHP Session 工作 机 制 流程 图 


9 


第 3 篇 ”PHP 编程 基础 


通过 上 面 这 张 流程 图 ,可 以 清楚 地 知道 PHP 的 会 话机 制 。 如 果 大 家 的 网 站 架设 在 共享 
服务 器 上 ， 且 访问 该 网 站 的 用 户 没 有 开通 Cookie 功能 ，( 车 非 通过 手动 的 方式 在 所 有 的 页 
面 中 传递 Session ID ) 整个 网 站 的 Session ID 就 有 可 能 失效 。 

现在 详细 地 讲 讲 如 何 创 建 Session、 传 递 Session ID、 存 取 Session 变量 及 销毁 Session 
的 内 容 。 


9.3.2 创建 及 销毁 Session 


只 有 在 每 个 页 面 上 都 开启 了 Session 功能 ，Session 变量 才 可 以 在 页 面 间 传递 。 开 启 
Session 功能 的 方法 如 下 : 


session _ start() 7 


这 条 语句 会 检查 现 有 的 Session ID .如 果 检 查 到 一 个 Session 的 存在 , 则 开始 创建 Session 
变量 ， 若 未 检查 到 任何 Session， 则 开始 创建 新 的 Session。 

需要 注意 的 是 ，session_start() 方 法 与 set_cookie() 的 方法 一 样 ， 都 必须 在 PHP 向 客户 端 
页 面 输出 任何 信息 之 前 执行 ， 否 则 Session 创建 失败 。 因 此 ， 最 好 的 方法 就 是 在 每 个 页 面 
的 开始 就 执行 session_start() 和 set_cookie() 方 法 。 

如 果 觉 得 这 种 办 法 有 些 麻烦 ,而 你 又 可 以 修改 服务 器 上 的 php.ini 文件 。 那么 在 该 文件 
中 查找 session_autostart 变量 ， 并 将 它 的 值 修改 成 |。 这样 ， 每 当 用 户 访 问 服务 器 上 的 任意 
页 面 时 ， 都 会 自动 执行 session_start() 方 法 。 

另外 ， 也 可 以 在 用 户 可 以 访问 某 些 页 面 之 前 对 它们 进行 身份 验证 : 只 有 通过 身份 验证 
的 用 户 才 允许 其 访问 这 些 页 面 。 那 么 就 需要 用 户 输入 用 户 名 和 密码 等 元 素来 登录 网 站 。 当 
用 户 离开 网 站 时 ， 也 需要 其 退出 。 关 于 登录 ， 我 们 可 以 使 用 session_start() 方 法 ， 而 对 于 退 
出 ， 可 以 使 用 下 面 这 条 语句 : 

session destroy () 

这 条 语句 执行 后 ， 当 前 Session 文件 内 的 所 有 变量 都 被 重 置 。PHP 也 不 再 向 下 一 个 页 
面 传递 Session ID。 但 是 执行 该 语句 的 页 面 不 会 受 此 影响 。 如 果 你 需要 将 当前 页 面 中 的 所 
有 Session 变量 都 失效 ， 则 需要 使 用 类 似 如 下 的 语句 : 


unset ($variableNamel, [$variableName2],...) 


9.3.3 使 用 Session 变量 


在 9.3.1 小 节 中 ， 我 们 提 到 了 Session 变量 这 个 词 。 所 谓 的 Session 变量 指 的 是 存储 在 
Session 文件 中 ， 供 后 一 次 会 话 期 间 的 所 有 文件 调用 的 变量 。 若 需要 在 后 续 页 面 中 调用 某 些 
在 当前 页 面 中 设置 的 变量 ， 可 以 在 当前 页 面 中 将 这 些 变量 以 键 值 对 的 方式 存 入 $_SESSION 
数组 中 ， 如 : 

$_SESSION['varname'] = "John Smith"; 

在 同一 会 话 期 间 的 后 续 页 面 中 ， 可 以 像 调 用 普通 数组 一 样 调用 这 个 数组 变量 。 

如 果 需 要 销毁 该 Session 变量 ， 需 要 使 用 unset0 方 法 ， 如 : 


unset ($_ SESSION[ "varname '"]) 7 
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在 例 9.5 中 将 定义 两 个 页 面 Ex9-04aphp 和 Ex9-04b.php 来 展示 如 何 使 用 Session 在 两 
个 页 面 间 传递 数据 。 
【 例 9.5】 使 用 Session 在 页 面 间 传 递 数 据 。 


下 


20 
el 


<?php 
session start(); 
$ SESSION["session var"] = "session testing"; 
?> 
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" 
"http://www.w3.org/TR/html4/loo0se.dtd"> 
<meta content = "text/html; charset=UTF-8" http-equiv="Content-Type"/> 
<html> 


<head> 
<title> 
Session Test Page 1 
</title> 
<body> 
<p>This is a test for data transmission through sessions</p> 
<form action="Ex9-04b.php" method = "post"> 
<input type="text" name="form var" value="form 
testing"/> 
<input type="submit" value="Go to Next Page" /> 
</form> 
</body> 
</head> 
</html> 


上 面 这 一 段 脚 本 使 用 session_start() 方 法 开启 了 Session 功能 , 接着 , 定义 了 一 个 Session 
变量 $ SESSION['session_var]， 其 值 为 session testing。 然 后 ， 用 HTML 定义 了 一 张 表单 ， 
定义 了 Ex9-04b.php 为 处 理 该 表单 的 动作 ， 并 规定 使 用 POST 方法 传递 数据 。 表 单 中 有 
个 文本 框 ， 名 为 form_var， 值 为 form testing。 

现在 来 写 Ex9-04b.php 文件 。 
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<?php 
session start(); 
2> 
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" 
"http://www.w3.o0org/TR/html4/loose.dtd"> 
<meta content="text/html; charset=UTF-8" http-equiv="Content-Type"> 
<html> 


<head> 
<title> 
Session Test Page 2 
</title> 
</head> 
<body> 
<?php 
$session var = $ SESSION['session_ var']; 
$form var = $ POST['form var']; 
echo "session var is $session var.<br>"; 
echo "form var is $form Var- "7 
人 
</body> 
</html> 


在 如 Ex9-04b.php 所 示 的 这 段 脚本 中 ， 我 们 在 页 面 的 开头 也 使 用 了 session_start0 方 法 
开启 了 Session 功能 。 接 着 ， 在 HTML 的 body 标签 下 ， 定 义 了 两 个 普通 变量 $session_var 


ss 
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和 $form var 用 来 存储 会 话 变量 和 由 表单 传递 过 来 的 变量 ， 它 们 的 值 分 别 为 
$_SESSION['session var] 和 $_POST[form_ var]。 

接着 运行 Ex9-04aphp， 然 后 单 击 “Go to Next Page” 按 钮 ， 会 发 现 页 面 显示 了 如 下 的 
两 段 话 : 

session var is session testing. 

form var is form testing. 


在 例 9.5 所 示 中 ， 我们 默认 用 户 开 启 了 Cookies 功能 ， 以 便 PHP 在 页 面 间 传 递 Session 
ID。 按 照 图 9-9 所 示 的 流程 图 来 看 ， 如 果 用 户 关 闭 了 浏览 器 的 Cookies 功能 。 系 统 会 检测 
服务 器 的 PHP 引擎 上 是 否 开启 了 trans_id 功能 。 若 已 开启 ，PHP 引擎 会 做 如 下 的 工作 : 
口 设 定 一 个 常量 ， 其 名 为 SID， 值 为 一 个 形 如 PHPSESSID=1ongstringofrumber 的 键 
值 对 。 
口 然后 在 一 个 用 户 会 话 期 间 ，PHP 引擎 会 主动 将 该 常量 的 值 ， 也 就 是 当前 Session 的 
ID 传递 到 用 户 访 问 的 每 个 页 面 。 
当然 ， 上 面 这 个 过 程 对 用 户 来 说 是 透明 的 。 也 就 是 说 ， 用 户 不 会 感知 到 使 用 Cookies 
传递 Session ID 和 使 用 SID 常量 配合 trans_id 传递 Session ID 的 差别 ,但 是 ,此 时 Session ID 
是 以 明文 的 方式 附加 在 URL 后 面 的 。 任何 人 都 可 以 看 到 该 Session ID, 容易 造成 安全 问题 。 
车 用 户 在 浏览 器 中 收藏 的 页 面 URL 中 带 有 Session ID， 那 么 当 用 户 单 击 收藏 夹 中 收藏 的 地 
址 访问 该 页 面 时 ， 新 的 Session ID 和 旧 的 Session ID 会 发 生 冲 突 ， 进 而 引发 一 系列 问题 。 
也 正 是 由 于 该 原因 ，PHP 引擎 中 的 trans id 功能 默认 是 禁用 的 。 
那么 如 果 用 户 既 没有 开启 Cookies， 服 务 器 上 也 没有 开启 trans_id 功能 ， 我 们 就 需要 在 
页 面 间 手动 传递 Session ID 了 。 
在 手动 传递 Session ID 时 ， 还 是 会 使 用 SID 常量 。 我 们 需要 将 其 手动 附加 到 需要 访问 
页 面 的 地 址 后 面 ， 如 : 


<a href = "nextpage.php?<?php echo SID?>" >Next Page</a> 


假设 当前 会 话 的 Session ID 为 877c22163d8df9deb342c7333cfe38a7， 那 么 常量 SID 的 
值 就 是 PHPSESSID=877c22163d8df9deb342c7333cfe38a7, 上 面 这 条 语句 指向 的 页 面 地 址 就 
应 该 是 nextpage.php? PHPSESSIONID=877c22163d8df9deb342c7333cfe38a7。 

这 样 一 来 ，Session ID 还 是 以 明文 的 方式 在 页 面 间 传 递 ， 会 出 现 很 多 的 问题 。 为 了 避 
免 这 种 情况 ， 我 们 也 可 以 通过 表单 的 隐藏 文本 框 来 帮助 我 们 传递 Session ID， 如 : 


1 <?php 

从 SPHPSESSID = session id() 

3 echo "<form action='nextpage.php' method='post'> 

4 <input type="hidden' name="'PHPSESSID' value="'$PHPSESSID'> 
5 <input type="'submit' value='Next Page'> 

6 </form>"; 

1 > 


上 面 这 条 语句 使 用 了 session_id0 函 数 , 获取 当前 Session 的 ID, 然后 定义 了 一 张 表 单 ， 
并 将 获取 到 的 Session ID 填 入 了 表单 中 的 隐藏 文本 框 内 。 当 用 户 单 击 “Next Page” 按 钮 时 ， 
Session ID 就 会 传递 到 下 一 个 页 面 中 。 读者 也 可 以 使 用 JavaScript 脚本 将 一 个 超 链接 转换 为 
提交 按钮 实现 通过 单 击 链接 提交 Session ID 的 功能 。 
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9.4 使 用 表单 上 传 文件 


我 们 除了 可 以 在 页 面 间 传递 数据 外 ， 还 可 以 向 服务 器 上 传 文件 。 例 如 ， 在 许多 的 论坛 


9.4.1 


里 ， 如 果 用 户 需要 上 传 文件 的 话 ， 就 需要 我 们 提供 一 个 上 传 文件 的 接口 。 


使 用 表单 上 传 文件 


首先 , 还 是 依 萌 芦 画 桔 , 先 看 看 其 他 的 网 站 上 是 如 何 做 文件 上 传 这 个 功能 的 , 如 图 9-10 


所 示 。 
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和 @ 文件 尺寸 小 于 500KB ,可 用 扩 霸 各 : mr rp,gfjpg pn 
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图 9-10 上 传 附件 


在 图 9-10 所 示 中 可 以 看 到 , 该 网 站 上 的 文件 上 传 组 件 是 由 两 个 按钮 构成 的 , 一 个 是 “ 选 
择 上 传 文件 ”按钮 ， 另 一 个 是 “确定 ”按钮 。 在 HIML 中 ， 原 生 的 上 传 组 件 是 通过 类 型 为 
“file” 的 表单 项 定义 的 ， 具 体 如 例 9.6 所 示 。 

【 例 9.6】 文件 上 传 表单 。 
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<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" 
"http://www.w3.o0org/TR/html4/loose.dtd"> 
<meta content="text/html; charset=UTF-8" http-equiv="Content Type"> 
<html> 
<head> 
<title>Upload a file</title> 
</head> 
<body> 
<p>You can click the button to select the file you want to 
upload:</p> 
<form enctype="multipart/form-data" action="Ex9-05b.php" 
method="post"> 
<input type="hidden" name="MAX FILE SIZE" value="30000"> 
<input type="file" name="user file"> 
<br> 
<input type="submit" value="Upload File"> 
</form> 
</body> 
</html> 
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上 面 这 段 HTML 代码 运行 后 ， 显 示 的 内 容 如 图 9-11 所 示 。 
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You can click the button to select the file you want to upload: 
Upload File 


图 9-11 HTML 原生 的 文件 上 传 组 件 


通过 图 9-11 所 示 可 以 看 到 , HTML 原生 的 上 传 组 件 是 由 一 个 文件 框 和 一 个 “Browse...” 
按钮 构成 的 。 图 9-11 所 示 中 的 两 个 按钮 “Browse...” 和 “Upload File” 就 相当 于 图 9-10 
所 示 中 的 “选择 上 传 文件 ”和 “确定 ”按钮 。 

关于 例 9.6 中 的 脚本 ， 还 有 两 点 需要 说 明 : 

口 如 果 需 要 通过 表单 上 传 文件 ， 那 么 表单 的 “enctype ”属性 必须 设置 为 


“multipart/form-data”， 耕 则 文件 上 传 后 可 能 会 出 现 错误 。 


口 另外 ， 表单 中 使 用 了 一 个 名 为 “MAX_FILE_SIZE” 的 隐藏 文本 框 ， 用 于 指定 用 户 


可 以 上 传 文件 体积 的 最 大 值 。 在 设置 这 个 值 的 时 候 ， 需 要 考虑 PHP 引擎 的 配置 文 
件 php.ini 中 两 个 变量 的 取 值 问题 。 一 是 “upload _ max filesize”， 另 一 个 是 
“post_max_size”。 前 者 是 允许 上 传 的 最 大 文件 大 小 ， 后 者 是 通过 POST 方法 允 
许 上 传 的 最 大 文件 大 小 。 注意 , 如 果 使 用 post 方法 上 传 文件 (我们 一 般 都 这 么 做 )， 
那么 upload_max_filesize 的 大 小 不 能 大 于 poast_max_size， 和 否则 一 定 会 出 现 问题 。 

如 果 你 从 来 没有 修改 过 php.ini 文件 中 这 两 个 参数 的 值 , 前 者 的 默认 值 为 2 MB, 后 
者 的 默认 值 为 8 MB。 


当 用 户 使 用 浏览 器 打开 Ex9-5.php 文件 、 选 中 某 个 文件 并 单 击 “Upload File” 按 钮 后 ， 
用 户 选 中 的 文件 被 上 传 到 一 个 用 于 存放 临时 文件 的 文件 夹 内 。 我 们 需要 编写 脚本 将 其 复制 
到 某 个 可 永久 存放 文件 的 文件 夹 内 ， 以 防 文件 在 脚本 执行 结束 后 被 删除 。 如 果 不 知道 你 的 
服务 器 使 用 的 临时 文件 夹 ， 可 以 使 用 phpinfo() 方 法 查看 。 


. 194 . 


ea 


-lxjsssss 2.] 


Eeroram FIESWESSYFHF- 
row 

ae 

印 

Er 


图 9-12 保存 用 户 上 传 文件 的 临时 文件 夹 
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按照 图 9-12 所 示 ， 使 用 的 服务 器 保存 用 户 上 传 文件 的 临时 文件 夹 为 “C:\Program 
Files\EasyPHP-12\imp ”。 如果 需要 修改 此 文件 夹 的 路 径 ， 可 以 打开 php.ini 文件 ， 搜 索 
“upload tmp _ dir” 参数 修改 。 

通常 情况 下 ， 临 时 文件 存放 在 哪里 并 不 重要 ， 一 般 不 需要 修改 php.ini 文件 中 
“upload tmp _ dir” 的 值 。 


9.4.2 ”获取 已 上 传 文件 的 信息 


在 用 户 上 传 某 个 文件 后 , 文件 会 被 存放 在 一 个 临时 文件 夹 内 。 这 时 , 我 们 可 以 使 用 PHP 
脚本 的 $_FILES 数组 来 处 理 这 个 文件 。 注 意 $_FILES 是 个 二 维 数组 : 第 一 个 维度 是 表单 中 
文件 域 的 名 字 ， 而 第 二 个 维度 则 是 文件 的 相关 属性 。 以 例 9.6 中 的 表单 所 示 ， 我 们 可 以 通 
过 以 下 的 方式 获取 已 上 传 文件 的 相关 属性 : 

echo $ FILES['user file']['name']; 

echo $ FILES['user file']['type']; 

echo $ FILES['user file']['tmp name']; 

echo $ FILES['user file']['size']; 

在 获取 文件 的 相关 属性 后 ， 就 可 以 判断 用 户 上 传 的 文件 是 否 符合 要 求 。 如 果 用 户 上 传 
的 文件 符合 我 们 的 要 求 ， 就 可 以 使 用 move_uploaded file0 函 数 把 文件 转移 到 永久 存放 文件 
的 文件 夹 内 。 

还 是 以 例 9.6 中 所 示 的 表单 为 例 ， 若 用 户 上 传 的 文件 名 为 testfile.txt， 那 么 我 们 就 可 以 
使 用 如 下 的 脚本 将 其 移动 到 永久 存放 文件 的 文件 夹 〈C:data\) 内 ， 并 将 其 重 命名 为 
new_file.txt: 

move_ uploaded file($ FILES['user file']['tmp name'],'C:\data\new file.txt'); 

需要 注意 的 是 ， 这 个 永久 存放 文件 的 文件 夹 在 存放 文件 之 前 必须 已 经 存在 ， 否 则 ， 上 
传 文件 失败 。 

下 面 ， 我 们 来 看 一 个 完整 的 例子 。 在 这 个 例子 中 ， 需 要 定义 一 个 表单 用 于 上 传 图 片 。 
在 用 户 单 击 “ 上 传 图 片 ” 按 钮 后 。 允 许 上 传 文件 最 大 不 能 大 于 500 KB。 系 统 会 对 用 户 上 传 
的 文件 进行 检测 ， 若 文件 大 小 超出 大 小 限制 或 文件 类 型 不 正确 ， 均 显示 上 传 失 败 。 若 文件 
大 小 和 文件 类 型 都 符合 要 求 ， 则 上 传 成 功 。 

【 例 9.7】 文件 上 传 : 一 个 完整 的 例子 。 


二 <?php 

if(!isset($ POST['UPload'])){ 

3 include ("Ex9-06a.php"); 

4 } else { 

5 if ($ FILES['image']['tmp name'] == "") { 

6 echo "<b>Failed to upload the image. Check the file size. 

六 File must be less than 800 KB.</b>"; 

8 include ("Ex9-06a.php"); 

9 exit(); 

10 } 

lt if (!preg match("/image/", $ FILES['image']['type’'])) { 

12 echo "<b>The type of the current file is ".$ FILES['image'] 
['type']." and is not an image. 

13 Please select another one.</b>"; 

14 include ("Ex9-06a.php"); 
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5 exit(); 

16 下 

Ky else { 

18 Sdestination = dirname( FILE )." \image\\".$ FILES 
['image']['name']; 

19 Stemp file = $ FILES['image']['tmp name"]7 

20 move uploaded file($temp file, $destination); 

有 2 echo "<p><b>Succeeded in uploading the image:</b></p> 

22 <p>{$ FILES['image']['name']} 22 ({$ FILES['image'] 

Pasize lI</p>"> 

23 } 

24 } 

Cs 


在 这 段 脚本 中 ， 我 们 需要 注意 的 内 容 如 下 所 示 。 

(1) 使 用 isset0 函 数 来 判断 5_POST['Upload"] 是 否 已 经 赋值 ， 这 里 的 “Upload” 为 表单 
按钮 的 名 称 。 如果 用 户 单 击 过 该 按钮 ， 那么 9$_POST['Upload"] 的 值 应 该 为 定义 该 按钮 时 指定 
的 值 。 

如 果 $_POST['Upload"] 的 值 还 没有 指定 , 也 就 是 用 户 从 来 就 没有 单 击 过 该 按钮 , 那么 系 
统 就 会 使 用 includeO 函 数 加 载 表 单 ， 提 示 用 户 选择 需要 上 传 的 文件 。 如 果 用 户 已 经 单 击 过 
该 按钮 ， 那么 $_POST['Upload"] 的 值 就 已 经 设置 , 系统 就 开始 判断 用 户 指 定 的 文件 是 否 符合 
要 求 。 

(2) 另外 ， 脚 本 里 使 用 了 两 个 站 条 件 来 判断 用 户 上 传 文件 的 类 型 和 大 小 。 若 其 中 一 项 
不 符合 要 求 ， 系 统 就 会 执行 exit0 函 数 退 出 脚本 的 执行 。 若 两 面 都 符合 要 求 ， 则 开始 上 传 文件 。 

在 上 传 文件 之 前 定义 了 两 个 变量 : $destination 和 $temp_file, 前 者 为 文件 上 传 后 的 永久 
存储 路 径 ， 后 者 为 文件 上 传 后 的 临时 存储 路 径 。 

(3) 在 定义 文件 上 传 后 的 永久 存储 路 径 时 ,我 们 使 用 了 dimame(_FILE_) 函 数 来 获取 
当前 文件 所 处 文件 夹 的 路 径 。 然 后 再 加 上 专门 用 于 存放 图 片 的 文件 夹 的 名 称 。 

需要 注意 的 是 ， 路 径 中 使 用 的 分 隔 符 为 “\”， 与 PHP 中 的 转 义 字符 相同 。 当 我 们 在 
字符 串 中 使 用 “\” 时 ， 该 符号 后 的 内 容 会 被 转 回 本 义 。 因 此 ， 我 们 在 专门 用 于 存放 图 片 的 
文件 夹 的 名 称 后 面 加 上 了 两 个 “\”， 前 一 个 为 转 义 字符 ， 后 一 个 为 路 径 分 隔 符 。 

现在 来 看 看 例 9.7 中 引用 的 表单 文件 : 


1 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" 

2 "http://www.w3.org/TR/html14/1oose.-dtd"> 

<meta content="text/html; charset=UTF-8" http-equiv="Content-Type"> 
4 <html> 

3 <head> 

6 <title>Upload Image</title> 

</head> 

8 <body> 

9 <ol> 

10 <li>Select the image you want to share with us.</l1i> 

2 <li>Click "Upload Image" button below after the 

1 path of the file appears in the text box.</l1i> 

3 </ol> 

14 <div align="center"> 

让 <hr> 

16 <form enctype="multipart/form-data" action="Ex9-06b.php" 
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method="POST"> 


Ey <input type="hidden" name="MAX FILE SIZE" value= 
"800000"> 

18 <input type="file" name="image" size="60"> 

19 <p></p> 

20 <input type="submit" name="Upload" value="Upload 
Image"> 

2 </form> 

22 </div> 

芝 袜 | </body> 

24 </html> 


在 这 份 文件 中 ， 我 们 定义 的 表单 中 文件 域 的 名 称 为 “image ”， 提 交 按 钮 的 名 称 为 
“Upload”， 其 值 为 “Upload Image”。 


9.5 实战 练习 : 记 账 工具 (中 ) 


在 上 一 章 的 实战 练习 中 ， 我 们 创建 了 一 个 bookKeeping 类 ， 用 于 逐条 或 批量 添加 收入 
和 消费 记录 。 在 本 章 里 ， 我 们 将 规划 记 账 工具 的 用 户 界面 和 数据 传递 方式 。 


9.5.1 界面 预览 


首先 ， 来 看 看 登录 界面 。 在 登录 界面 里 ， 我 们 提供 了 两 个 文本 框 : 一 个 让 用 户 输入 用 
户 名 ， 另 一 个 让 用 户 输入 密码 。 然 后 ， 还 提供 了 一 个 按钮 ， 可 以 让 用 户 提交 登录 。 

在 用 户 登录 系统 之 后 会 直接 进入 添加 支出 记录 的 页 面 。 在 该 页 面 的 左 侧 是 导航 栏 ， 用 
于 引导 用 户 分 别 进行 “添加 支出 记录 ”、“ 添 加 收入 记录 ”、“ 批 量 上 传记 录 ” 和 “查看 
记录 列表 ”; 而 在 页 面 的 右 侧 则 是 功能 区 ， 用 户 可 以 通过 填写 表单 完成 相应 的 任务 ， 如 图 
9-13 一 图 9-17 所 示 。 


Book Keeping 


图 9-13 ”登录 页 面 


"Ts 
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添加 一 条 消费 记录 


图 9-14 添加 消费 记录 


添加 一 条 收入 记录 


图 9-15 添加 收入 记录 


图 9-16 批量 导入 记录 
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欢迎 您 , tobunka 


添加 滑 旨 记录 查看 记录 列表 

天 jn 和 记录 EE [Ez 

批量 导入 记录 

Em E 


图 9-17 查看 记录 列表 


9.5.2 脚本 分 析 


这 几 个 页 面 的 HIML 部 分 , 我 们 就 不 在 书 中 展示 了 , 大 家 可 以 去 看 脚本 。 不 过 在 这 里 ， 
还 是 需要 分 析 一 下 登录 这 个 过 程 的 。 所 谓 登录 ， 就 是 指 用 户 在 输入 了 正确 的 用 户 名 和 密码 
之 后 ， 可 以 获得 授权 ， 访 问 相应 的 页 面 。 在 这 里 ， 我 们 需要 让 用 户 在 成 功 登 录 后 ， 可 以 访 
问 如 图 9-13 到 图 9-17 所 示 的 页 面 。 那 么 在 这 里 ， 就 要 用 上 Session 功能 。 

在 登录 页 面 上 ，PHP 脚本 如 下 : 


<?php 
session start(); 
if(isset($ REQUEST['login']))t{ 
Suser = $ REQUEST['usr']; 
Spws = $ REQUEST['psw']; 


if(strlen($user) > 0) { 
$_SESSION['user'] = $user; 
header ('location:./08.2 Book Keeping AddExpense.php'); 
bh 
} 
a 


在 上 面 这 段 脚本 中 ， 首 先 使 用 “session_start()” 启 用 了 Session 功能 。 然 后 再 判断 用 户 
是 否 提 交 了 表单 。 车 是 ， 则 获取 用 户 提 交 的 用 户 名 和 密码 ， 并 对 其 进行 判断 。 在 这 里 ， 为 
了 简化 代码 , 我 们 只 判断 用 户 是 否 输 入 了 用 户 名 , 并 据 此 决定 是 否 将 用 户 转 向 指定 的 页 面 。 

脚本 中 使 用 了 header0 函 数 ， 用 来 将 用 户 转向 指定 的 页 面 。 需 要 注意 的 是 ， 在 使 用 
header0) 函 数 之 前 ， 不 得 向 浏览 器 输出 任何 数据 ， 否 则 header 的 转向 功能 不 会 起 作用 。 

接 下 来 再 看 看 其 他 四 个 页 面 中 的 与 Session 相关 的 PHP 脚本 吧 : 


<?php 
session start(); 
include('07 Book Keeping Class.php'); 
if(isset($ SESSION['user']))1{ 

2 
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/ * HTML 代码 已 略 去 * / 
<?php 
} else { 
header ('location:./08.1 Book Keeping Login.php'); 
} 


全 之 


上 面 这 段 脚本 同样 也 使 用 “session_start() ”函数 启用 了 Session 功能。 然后 判断 名 为 
“user” 的 Session 变量 。 若 存在 ， 则 向 浏览 器 输出 上 面 略 去 的 HTML 代码 ;反之 ， 则 将 用 
户 转 到 登录 页 面 。 

最 后 ， 在 批量 导入 记录 时 ， 需 要 上 传 模板 文件 。 在 下 一 章 里 将 学 习 如 何 读 取 文 件 中 的 
内 容 ， 因 此 ， 制 订 模板 文件 我 们 放 在 下 一 章 的 实战 练习 中 讲解 。 在 这 一 章 里 ， 只 需要 将 用 
户 填写 好 的 模板 文件 上 传 到 服务 器 上 即 可 。 

在 9.4 节 里 ， 我 们 学 习 了 如 何 使 用 表单 上 传 文件 到 服务 器 ， 现 在 我 们 在 “批量 导入 记 
录 ” 的 页 面 上 添加 相应 的 脚本 来 获取 上 传 后 的 文件 : 

<?php 


// 判断 用 户 是 否 已 提交 表单 
if(isset($ _ REQUEST["batchUpload']))1{ 

// 获取 用 户 上 传 的 文件 信息 

$temp name = $ FILES['upload'] ['tmp name']; 

Sname = $ FILES['upload']['name']; 

$typeof = $ FILES['upload']['type']; 

pe 于 

if($temp name == "" 

!preg match("' ph ,$typeof)){ 
Smsg = “上传 失败 或 格式 不 正确 。 


} else { 
// 移动 文件 到 指定 目录 
$destination = "upload'7 


move_ uploaded file($tmp name, '$destination/$name'); 
// 使 用 自 定义 类 的 readTemp () 方法 来 处 理 模板 文件 〈 见 第 10 章 ) 
Smsg = readTemp (Sname) 


ji 
?> 
至 此 ， 完 成 了 为 记 账 工具 搭建 可 用 界面 的 任务 。 在 第 10 章 里 ， 我 们 将 为 页 面 添加 相 
应 的 功能 


9.6 习 题 


(1) 请 编写 一 个 HTML 页 面 ， 用 于 收集 用 户 的 身份 信息 。 主 要 包括 : 姓名 、 性 别 、 出 
生年 月 以 及 照片 。 

(2) 请 编写 一 个 PHP 页 面 ， 用 于 处 理 用 户 通过 表单 提交 的 信息 。 在 该 页 面 上 引用 名 为 
class.authenticate .php 的 类 文件 。 

(3) 请 编写 名 为 class.authenticate .php 的 类 文件 ， 用 于 验证 用 户 提交 的 信息 。 
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在 第 9 章 里 ， 我 们 学 习 了 如 何 使 用 表单 从 用 户 那 里 收集 数据 、 如 何 使 用 URL 传递 数 
据 到 数据 处 理 页 面 、 如 何 使 用 Cookie 缓存 数据 以 及 如 何 使 用 Session 保证 数据 传输 过 程 的 
安全 。 但 似乎 还 缺少 了 很 重要 的 一 步 ， 那 就 是 如 何 将 这 些 传 递 来 的 数据 储存 起 来 供 以 后 使 
用 呢 ? 

数据 库 的 功能 很 强大 ， 相 信 很 多 同学 都 听 说 过 数据 库 。 只 不 过 这 个 名 字 听 上 去 就 很 有 
距离 感 ， 以 至 于 很 多 听 说 过 它 的 人 也 不 见得 会 使 用 它 。 其 实 ，PHP 只 是 个 脚本 语言 ， 它 与 
我 们 用 什么 方式 存储 数据 没有 关系 。 我 们 可 以 使 用 PHP 脚本 语言 把 数据 存储 到 文本 文件 
中 ， 也 可 以 把 数据 以 结构 化 的 方式 存储 到 XML 文件 中 ， 当 然 也 可 以 把 在 需要 存储 的 数据 
之 间 建 立 关联 系 以 方便 我 们 把 数据 存储 到 关系 型 数据 库 中 。 

在 学 习 如 何 将 数据 存储 到 这 些 介 质 中 之 前 。 先 来 看 看 它们 各 自 都 有 些 什 么 优 缺 点 : 

(1) 对 于 文本 文件 来 说 ， 小 巧 轻便 应 该 是 决定 我 们 选择 文本 文件 的 关键 词 。 和 数据 库 
相 比 ， 一 个 文本 文件 在 存储 同样 多 数据 时 ， 占 用 的 磁盘 空间 更 小 ， 更 容易 被 读 取 。 但 是 需 
要 注意 的 是 它 不 能 存储 过 多 的 数据 ， 否 则 服务 器 在 文件 中 查找 起 来 的 速度 会 异常 的 缓慢 ， 
进而 会 影响 到 整个 站 点 的 服务 水 平 。 打 个 容易 理解 的 比方 ， 就 像 我 们 在 一 间 汗 牛 充 栋 的 书 
房 里 找 一 篇 文章 一 样 。 在 没有 任何 的 指引 情况 下 ， 就 只 能 对 其 展开 地 毯 式 搜索 ， 人 力 、 物 
力 耗费 都 很 大 。 

(2) 对 于 XML 文件 来 说 ， 结 构 化 应 该 是 决定 我 们 选择 它 的 关键 词 。 和 文本 文件 相 比 
-个 XML 文件 有 着 结构 化 的 表现 形式 ， 对 于 存储 在 其 中 数据 分 门 别 类 ， 有 条 有 理 。 虽 然 
XML 存储 同样 多 的 数据 时 , 会 比 文本 文件 占用 的 磁盘 空间 更 大 , 但 是 它 的 数据 结构 决定 了 
在 其 中 查找 数据 的 速度 。 就 好 像 我 们 把 刚才 那 间 书房 里 的 书 分 门 别 类 的 整理 好 了 ， 想 要 找 
一 本 编程 方面 的 书 ， 屠 就 只 管 按 图 索 台 查询 即 可 ， 肯 定 会 比 在 杂乱 无 章 的 书房 里 找 要 快 的 
多 。 不过， XML 文件 存储 的 数据 依然 不 能 过 量 ， 一 旦 过 量 ， 系 统 反 应 也 会 慢 下 来 ， 一 样 会 
影响 到 整个 站 点 的 服务 水 平 。 

(3) 对 于 关系 型 数据 库 来 说 ， 数 据 关系 应 该 是 决定 我 们 选择 它 的 关键 词 。 如 果 我 们 需 
要 存储 和 反复 使 用 的 数据 之 间 有 着 十 分 明显 的 逻辑 关系 ， 而 且 可 预见 的 数据 量 十 分 巨大 ， 
那么 最 好 使 用 关系 型 数据 库 ， 而 不 是 上 述 两 类 存储 介质 。 同 时 ， 数 据 库 一 般 会 有 强大 的 安 
全 能 力 ， 保 证 存储 在 其 中 的 数据 安全 。 这 一 点 也 是 上 述 两 种 介质 做 不 到 的 。 
虽然 文本 文件 和 XML 文件 有 些 缺 点 , 虽然 数据 库 有 着 文本 文件 和 XML 文件 无 法 比拟 
的 优点 ， 在 选择 存储 介质 时 ， 最 好 量体裁衣 ， 不 要 贪 大 求 洋 ， 以 免 造 成 浪费 。 


10.1 使 用 文本 文件 存 取 数据 


我 们 在 这 里 说 的 文本 文件 ， 顾 名 思 义 ， 指 的 是 只 包含 文本 的 文件 。 如 果 读 者 使 用 的 是 
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Windows 操作 系统 ， 那 么 选择 “所 有 程序 ” |“ 附件 ” |“ 记事 本 ”命令 ， 然 后 创建 一 个 文本 
文件 。 如 果 使 用 的 是 Unix 操作 系统 , 也 可 以 使 用 操作 系统 中 附带 的 类 似 于 记事 本 的 程序 创 
建文 本 文件 。 

在 我 们 的 印象 里 文本 文件 应 该 指 的 就 是 TXT 文件 , 这 种 看 法 比较 狭隘 。 其 实 文本 文件 
的 扩展 名 无 论 是 什么 ， 它 都 应 该 可 以 用 记事 本 或 与 之 类 似 的 程序 读 取 。 我 们 说 XML 文件 
也 是 一 种 文本 文件 ， 因 为 它 只 包含 文本 ， 而 且 可 以 用 记事 本 程序 来 读 取 。 在 本 节 里 ， 涉 及 
到 的 文本 文件 , 除了 TXT 文件 之 外 , 还 有 CSV 和 TSV 文件 , 它们 分 别 是 Comma Separated 
Values 和 Tab Separated Values 的 缩写 。 通 常用 在 不 同 的 应 用 程序 之 间 传 递 数据 。 

文本 文件 十 分 简单 易 用 , 不 需要 安装 任何 其 他 的 程序 , 就 可 以 用 PHP 脚本 来 读 写 文本 
文件 。 我 们 读 写 文本 文件 需要 分 三 步 走 : 第 一 步 是 打开 文本 文件 、 第 二 步 是 读 取 文 本 文件 
或 往 文本 文件 中 写 入 数据 、 第 三 步 则 是 关闭 文本 文件 。 这 三 个 步骤 虽然 看 上 去 和 “怎么 把 
一 头 大 象 放 在 冰箱 里 ”的 答案 如 出 一 略 ， 但 它们 缺 一 不 可 。 

下 面 来 看 看 如 何 用 PHP 脚本 打开 一 个 文本 文件 。 


10.1.1 打开 和 关闭 文本 文件 


按照 之 前 说 的 ， 在 读 写 文本 文件 之 前 ， 应 该 打开 这 个 文本 文件 。 我 们 可 以 使 用 如 下 的 
语句 打开 一 个 文本 文件 ， 并 将 这 个 文本 文件 存储 到 一 个 变量 中 。 

$fh = fopen("filename", "mode") 

在 上 面 这 条 语句 中 使 用 了 fopen0 函 数 。 它 带 有 两 个 必要 参数 ， 一 个 为 文件 名 ， 另 一 个 
为 模式 。 需 要 注意 的 是 ， 在 为 fopen() 函 数 添 加 参数 时 ， 一 定 要 用 引号 将 这 两 个 参数 分 别 包 

另外 ,在 打开 一 个 文本 文件 时 ， 一 定 要 指定 文件 的 打开 方式 。 表 10-1 所 示 详 细 地 罗列 
了 fopenO 函 数 支持 的 文本 文件 打开 方式 。 

表 10-1 fopen() 函 数 支持 的 文本 文件 打开 方式 


模式 | 打开 方式 说 ”了 明 

r 只 读 如 果 指定 的 文件 不 存在 ， 系 统 会 显示 一 条 警告 信息 

TI 读 写 如 果 指定 的 文件 不 存在 ， 系 统 会 显示 一 条 警告 信息 

国 只 写 如 果 指 定 的 文件 不 存在 ， PHP 会 尝试 创建 该 文件 。 如 果 指 定 的 文件 存在 , PHP 会 党 
试 覆 盖 该 文件 

wi | 读 写 如 果 指 定 的 文件 不 存在 ， PHP 会 尝试 创建 该 文件 。 如 果 指 定 的 文件 存在 ,PHP 会 党 
试 夏 盖 该 文件 

追加 如 果 指 定 的 文件 不 存在 , PHP 会 尝试 创建 该 文件 。 如 果 指定 的 文件 存在 ，PHP 会 尝 
试 在 此 文件 末尾 追加 数据 

a+ | 读 并 追加 和 如果 指定 的 文件 不 存在 ， PHP 会 尝试 创建 该 文件 。 如 果 指 定 的 文件 存在 , PHP 会 尝 
试 在 此 文件 末尾 追加 数据 


我 们 在 指定 文件 名 时 ， 可 以 只 使 用 文件 名 ， 前 提 是 该 文件 与 执行 读 取 该 文件 命令 的 脚 
本 文件 在 同一 个 目录 下 。 和 否则 请 指定 该 文件 的 路 径 ， 可 以 使 用 绝对 路 径 或 相对 路 径 。 如 果 
需要 读 取 的 文件 不 在 本 地 ， 也 可 以 使 用 该 文件 的 URL。 

现在 以 只 读 的 方式 打开 一 个 文本 文件 : 


“20" 
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Sth = fopen("file. txt", vr™)s 


如 果 系 统 找 不 到 文件 file.txt， 或 者 这 个 文件 根本 就 不 存在 ， 则 系统 会 输出 一 条 警告 信 
息 ， 就 像 下 面 这 个 信息 一 样 : 


Warning:fopen (file.txt) :failed to open stream:No such file or directory in 
C:\greatwall\chapter10\Ex10-01.php on line 2 
虽然 系统 给 出 了 警告 信息 ， 但 那 也 仅仅 是 一 条 警告 而 已 ， 不 代表 脚本 停止 运行 了 。 脚 
本 还 会 继续 运行 ， 只 是 如 果 后 续 的 脚本 中 对 有 该 文件 的 读 写 操作 ， 它 们 都 不 会 被 执行 。 

为 了 避免 这 种 情况 ， 可 以 在 打开 该 文本 文件 之 前 ， 使 用 file_exists0 函 数 判 断 一 下 该 文 
件 是 否 存 在 。 因 此 ， 上 面 这 段 脚本 可 以 改 成 : 


王 <?php 

加 if (file exists("fileReadOonly.txt")) { 

3 $fh = fopen ("fileReadOnly.txt", "r"); 
4 } else { 

i die("The file does not exit."); 

6 UL 

学 3 


在 脚本 的 第 2 行使 用 了 file_existsO0 函 数 ， 它 只 有 一 个 必要 参数 ， 那 就 是 文件 名 。 如 果 
这 个 文件 在 当前 目录 中 存在 的 话 ， 那 么 就 会 以 文本 流 的 形式 存储 在 变量 $fth 中 ; 否则， 系 
统 会 输出 一 条 我 们 定义 的 信息 。 

现在 ， 我 们 再 以 只 写 的 方式 打开 一 个 文件 。 需 要 注意 的 是 ， 当 我 们 以 只 写 的 方式 打开 
一 个 文本 文件 时 ， 如 果 文 件 存在 ， 该 文件 被 覆盖 ， 如 果 文 件 不 存在 ，PHP 会 创建 该 文件 。 

$fh = fopen ("fileWriteOnly.txt", "w"); 

运行 这 段 脚 本 后 会 发 现 ， 当 前 目录 下 出 现 了 一 个 新 的 文本 文件 。 该 文本 文件 的 名 字 正 
好 是 我 们 在 脚本 中 定义 的 fleWriteOnly.txt。 

假 车 当前 目录 下 有 一 个 子 目 录 为 subfolder。 我 们 可 以 使 用 如 下 的 脚本 以 只 写 的 方式 打 
开 该 子 目 录 下 的 文本 文件 : 

$fh = fopen("subfolder/fileWriteOnly.txt", "w"); 


当然 , 如 果 当 前 目录 下 并 没有 一 个 名 为 subfolder 的 子 目 录 , 则 会 出 现 如 下 的 报错 信息 : 


Warning: fopen (subfolder/fileWriteOnly.txt): failed to open stream: No such 
file or directory in C:\greatwall\chapter10\Ex10-01.php on line 8 


因此 , 在 读 取 子 目 录 中 的 文本 文件 之 前 , 还 是 需要 判断 一 下 该 子 日 录 是 否 存 在 。 这 里 
我 们 可 以 用 到 is_dir0 函 数 。 上 面 这 段 脚 本 ， 就 可 以 改 成 : 


时 <?php 

世 if (is dir("subfolder")) { 

3 $fh = fopen("subfolder/fileWriteOnly.txt", "w"); 
4 } else { 

5 die("The directory does not exist."); 

6 

by 


} 


> 


现在 ,我们 再 来 打开 一 个 远程 的 文本 文件 。 所 谓 远程 的 文本 文件 ， 也 就 是 说 该 文本 文 
件 不 在 本 地 。 因 此 ， 建 议 读 者 以 只 读 方式 打开 ， 并 对 该 文件 做 读 写 操作 。 除 非 你 确信 你 对 
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该 文件 有 读 写 权限 。 

$fh=fopen ("http://www.gutenberg.org/cache/epub/42304/pg42304.txt", "Fr") 7 

上 面 这 段 脚 本 打开 了 位 于 古 腾 保 计划 官方 网 站 上 的 一 本 TXT 格式 的 免费 电子 书 《 日 本 
之 密 》 (The Gist of Japan) 。 

如 果 运 行 这 段 脚本 ， 没 有 系统 报错 的 话 ， 那 么 恭喜 你 成 功 打 开 了 这 本 电子 书 。 

在 学 会 如 何 打开 文本 文件 之 后 , 现在 要 来 看 看 如 何 关闭 打开 的 文本 文件 。 当 使 用 fopen0 
函数 打开 某 个 文本 文件 之 后 ， 该 文本 文件 就 以 文件 流 的 形式 存放 在 某 个 变量 中 。 如 果 需 要 
关闭 这 个 文件 流 ， 我 们 可 以 使 用 fcloseO 函 数 。 就 像 下 面 这 段 脚本 一 样 : 

fclose ($fh); 


在 上 面 这 段 脚本 中 ,使 用 的 fcloseO 函 数 只 有 一 个 必要 参数 ， 那 就 是 存储 了 已 经 打开 的 
文件 流 的 变量 。 

现在 总 结 一 下 本 节 中 使 用 到 的 脚本 。 

【 例 10.1】 打开 和 关闭 文本 文件 。 

出 <?php 


之 if (file exists("fileReadOonly.txt")) { 

3 $fh = fopen("fileReadOnly.txt", "r"); 
4 } else { 

die ("The file does not exit."); 

6 } 

辽 

8 /* 只 有 当 fileReadonly .txt 被 成 功 打开 ， 

9 下 面 的 脚本 才 会 被 运行 。*/ 

10 

El if (is dir("subfolder")) { 

了 又 $fh = fopen("subfolder/fileWriteOnly.txt", "w"); 
了 名 } else { 

14 die("The directory does not exist."); 
15 } 

16 


17 /* 只 有 当 subfolder 文件 夹 存 在 时 ， 
18 下 面 的 脚本 才 会 被 运行 。*/ 


19 

20 $fh= fopen ("http://www.gutenberg.org/cache/epub/42304/pg42304.txt", "r"); 
hl 

防 朗 fclose ($fh); 

Fe 


10.1.2 ”向 文本 文件 中 写 入 数据 


在 以 只 写 、 读 写 、 追 加 或 读 并 追加 的 方式 打开 了 一 个 文本 文件 之 后 ， 我 们 就 可 以 向 文 
本 文件 中 写 入 数据 了 。 在 向 文本 文件 中 写 数据 时 ， 会 用 到 fwrite0 函 数 。 

假如 ， 我 们 需要 向 某 个 文本 文件 中 追加 今天 的 日 期 。 我 们 可 以 这 么 写 ， 如 下 所 示 。 

【 例 10.2】 向 文本 文件 中 写 入 数据 。 


1 <?php 
部 S$today = date("Y-m-d"); 
和 $fh = fopen("fileAppendTo.txt", "a"); 
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4 fwrite($fh, $today."\n"); 

3 fclose ($fh); 

6 Es 

上 面 这 段 脚本 首先 使 用 “追加 ”的 方式 打开 了 文件 fleAppendTo.txt。 按 照 表 10-1 的 
说 明 ， 以 “追加 ”方式 打开 一 个 文本 文件 时 ， 若 文件 不 存在 ，PHP 会 尝试 创建 该 文件 。 然 


后 我 们 使 用 了 fwrite0 函 数 向 打开 的 文件 中 追加 今天 的 日 期 。 这 个 fwrite0 函 数 一 共 有 两 个 
必要 参数 ， 一 个 是 存储 着 以 文件 流 的 形式 打开 的 文本 文件 的 变量 ， 另 一 个 是 需要 追加 的 内 
容 。 在 追加 完成 后 ， 用 fclose0 函 数 关 闭 了 该 文件 流 。 

若 在 运行 这 段 脚 本 之 前 ，fleAppendTo.txt 不 存在 。 运 行 这 段 脚本 之 后 ， 会 发 现 当前 目 
录 下 多 了 一 个 名 为 fleAppendTo.txt 的 文件 。 打 开 这 个 文件 ， 里 面 赫然 记载 着 今天 的 日 期 。 
如 果 反 复 运 行 这 个 文件 ， 会 发 现 越 来 越 多 的 日 期 被 追加 到 了 文件 末尾 ， 就 像 下 面 这 样 : 


2013=03=12 
20E3=03=T2 


10.1.3 ”从 文本 文件 中 读 取 数 据 


在 向 文本 文件 中 写 入 了 数据 之 后 ， 我 们 的 诉求 就 变 得 很 明显 了 ， 那 就 是 把 这 些 数 据 读 
出 来 。 从 文本 文件 中 读 取 数 据 ， 可 以 用 到 fgets0 函 数 。 需 要 注意 的 是 ， 这 里 的 gets 是 第 三 
人 称 单数 形式 ， 不 是 原型 ， 另 外 ， 这 个 函数 只 有 一 个 必要 参数 ， 那 就 是 存储 着 以 文件 流 的 
形式 打开 的 文本 文件 的 变量 。 这 个 fgets0 函 数 的 返回 值 可 能 是 整个 文本 文件 ， 也 可 能 是 这 
个 文本 文件 的 某 一 行 。 到 底 是 整个 文本 文件 ， 还 是 这 个 文本 文件 的 某 一 行 ， 是 由 文本 文件 
中 的 换行 符 ， 也 就 是 在 例 10.2 的 fwrite0 函 数 中 使 用 的 “nm” 决定 的 ， 如 果 脚 本 在 读 取 数据 
时 碰 到 了 换行 符 ， 则 fgets0 返 回 就 是 该 换行 符 之 前 的 内 容 ， 如 果 脚 本 一 口气 读 到 了 文件 末 
尾 ， 那 么 fgets0 函 数 返 回 的 就 是 整个 文本 文件 的 内 容 了 。 

比如 ， 可 以 用 下 面 这 段 脚本 来 读 取 例 10.2 中 创建 的 fileAppendTo.txt 文件 。 

【 例 10.3】 从 文本 文件 中 读 取 数 据 。 


I <?php 

Es if (file exists("fileAppendTo.txt")) { 

3 $fh = fopen ("fileAppendTo.txt", "“r"); 
4 while (!feof($fh)) { 

5 $line = fgets ($fh); 

6 echo $line; 

7 } 

8 fclose ($fh) 

a } 

Oe> 


上 面 这 段 脚 本 先是 用 file_exists0 〇 函数 判断 fleAppendTo.txt 文件 是 否 存 在 。 若 该 文件 存 
在 于 当前 目录 下 ， 我 们 就 用 fopenO 函 数 以 只 读 的 方式 打开 这 个 文件 ， 并 将 该 文件 以 文件 流 
的 形式 存放 于 变量 $fth 中 。 然 后 使 用 while 语句 循环 输出 该 文件 每 一 行 的 内 容 。 

在 while 语句 中 , 我们 使 用 到 了 feofO 函 数 。 这 个 函数 是 用 来 判断 当前 指针 是 否 已 经 到 
达 文 件 末端 的 。 如 果 没有 ， 就 将 当前 行 的 内 容 输出 给 变量 $line， 然 后 输出 Sline 的 内 容 。 

上 面 这 段 脚本 的 运行 的 结果 如 下 : 

CURE Ek 


“es 
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有 的 同学 可 能 会 问 : 这 明明 是 两 行 的 内 容 ， 为 什么 会 显示 在 一 行 里 呢 ? 这 个 问题 在 之 
前 的 章节 中 已 经 讨论 过 了 ， 大 家 可 以 再 回想 一 下 ， 我 就 不 多 言 了 。 

现在 来 看 看 如 何 才 能 一 小 段 一 小 段 地 读 取 文本 文件 里 的 内 容 。 假 如 我 们 需要 4 个 字符 
为 一 行 地 显示 文本 文件 里 存储 的 内 容 ， 可 以 把 例 10.3 修改 成 如 下 的 样子 。 

【 例 10.4】 分 段 读 取 文本 内 容 。 

<?php 


if (file exists("fileAppendTo.txt")) { 

3 $fh = fopen ("fileAppendTo.txt", "r"); 
和 while (!feof($fh)) { 

5 Spiece = fgets ($fh, 5); 

6 echo "$piece<br>"; 

7 } 

8 fclose ($fh); 

9 } 

Oe 


在 上 面 这 段 脚 本 中 ，fegets() 函 数 指定 了 两 个 参数 。 一 个 为 存储 着 以 文件 流 的 形式 打开 
的 文本 文件 的 变量 ， 另 一 个 为 一 次 需要 读 取 的 字符 数 。 需 要 注意 的 是 ， 脚 本 在 实际 读 取 数 
据 时 返回 的 字符 数 会 比 指定 的 字符 数 少 一 位 。 因 此 ， 在 指定 一 次 需要 读 取 的 字符 数 时 ， 请 
务必 加 1。 
上 面 这 段 脚本 输出 的 结果 如 下 : 


2013 
上 全 刁 
12 
2013 
总 二 
12 


我 们 也 可 以 把 文本 文件 里 的 内 容 一 行 一 行 的 输出 到 一 个 数组 里 ， 就 像 下 面 这 样 。 
【 例 10.5】 将 文本 文件 中 的 数据 读 取 到 数组 中 。 


和 <?php 


2 if (file exists("fileAppendTo.txt")){ 
3 $fh = fopen ("fileAppendTo.txt", "r"); 
4 while(!feof ($fh)){ 

$line[] = fgets ($fh); 

6 } 

了 fclose ($fh); 

8 } 

9 

10 foreach ($line as $key => $value) { 
和 echo "[".$key."] ".$value."<br>"; 
12 } 

区 


如 果 大 家 还 记得 我 们 在 讲 数组 的 时 候 ， 提 到 的 数组 元 素 的 遍历 ， 那 么 应 该 对 foreach 
这 个 语句 有 些 印 象 。 在 这 里 先 把 fileAppendTo.txt 中 的 内 容 读 到 了 Sline 数组 中 存储 起 来 ， 
然后 用 foreach 语句 遍历 $line 数组 。 

上 面 这 段 脚 本 的 运行 结果 如 下 : 


LOT 20OES=03=1x 
1 20L3=03=12 
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如 果 我 们 需要 一 次 性 读 取 某 个 文本 文件 里 的 所 有 内 容 到 一 个 变量 中 ， 那 就 不 能 继续 使 
用 fgets0 函 数 了 。 这 时 ， 需 要 使 用 fie_get_contents() 函 数 ， 就 像 下 面 这 样 : 

$content = file get contents ("fileAppendTo.txt", 1); 

这 时 ，$content 的 值 就 是 一 个 字符 串 ， 而 不 是 一 个 数组 了 。 如 果 这 时 我 们 使 用 echo 语 
句 打印 变量 $content 的 值 ， 结 果 就 应 该 是 下 面 这 个 样子 : 

OE 


10.1.4 ”从 CSV 和 TSV 文件 中 读 取 数据 


文本 文件 存储 的 数据 除了 可 以 分 行 以 外 ， 似 乎 显得 有 些 杂乱 无 章 。 而 当 我 们 从 一 个 文 
本 文件 中 把 数据 读 取出 来 以 后 ， 需 要 把 它 转 存 到 另外 一 个 文件 中 时 ， 如 果 这 个 文件 对 导入 
的 数据 有 格式 上 的 要 求 ， 那 单纯 的 fppen0、fgets0 和 fclose0 函 数 就 有 点 力不从心 了 。 事 实 
上 ， 在 这 一 节 的 开头 ， 我 们 曾经 提 到 了 CSV 和 TSYV 文件 ， 知 道 它们 通常 会 被 用 来 在 不 同 
的 应 用 程序 间 传 递 数 据 。 根 据 这 一 点 可 以 推断 : PHP 应 该 会 提取 CSV 文件 或 TSV 文件 中 
的 数据 的 方法 。 

那么 什么 是 CSV 文件 呢 ? 

CSV 文件 是 一 种 有 着 简单 格式 的 文本 文件 。 顾 名 思 义 ， 这 种 类 型 的 文本 文件 里 存储 着 
用 逗号 分 隔 开 来 的 一 系列 的 值 ， 就 像 下 面 这 样 : 

John Smith, 1234 Oak St., London, GB, 999990 

Wang Xiaoer, A28 Shangdi Sanjie St., Beijing, PRC, 100085 

打开 记事 本 ， 将 上 面 这 两 行内 容 复制 到 记事 本 中 ， 然 后 选择 “文件 ”|“ 另 存 为 …” 命 
令 。 在 弹出 的 对 话 框 中 , 文件 格式 选择 “所 有 文件 ”然后 在 文件 名 中 输入 “addressbook.csv”， 
单 击 “ 保 在 ”按钮 。 一 个 CSV 文件 就 做 好 了 。 

如 果 你 的 电脑 上 还 安装 了 Microsoft Office 或 者 金山 的 WPS 这 样 的 办 公 软 件 ， 那 么 共 
喜 你 。 你 还 可 以 用 它们 的 电子 表格 软件 打开 这 个 后 级 名 为 CSV 的 文件 ， 如 图 10-1 所 示 。 


ELERED = 国 addressbook.csov* 器 | 
-| ma  John Snith 
B 


| We 


[Iobn Smith _] 1234 nak st. London | 6B 
Wang Xiaoer | A28 Shangdi Sanjie St,， Feijing ，PRC 100085 


图 10-1 用 电子 表格 软件 打开 后 绥 名 为 CSV 的 文本 文件 
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我 们 也 可 以 用 例 10.6 中 的 脚本 来 创建 一 个 CSV 文件 。 
【 例 10.6】 用 PHP 创建 CSV 文件 。 
村 <?php 


区 /* 首先 我 们 以 追加 模式 打开 一 个 文件 ， 以 防 

3 该 文件 在 当前 目录 下 不 存在 */ 

4 $fh = fopen("fileCSV.csv", "a"); 

5 

6 /* 然后 我 们 可 以 创建 一 个 数组 ， 用 来 存放 那些 

7 需要 存放 到 该 CSV 文件 中 的 数据 */ 

8 $address[] = "John Smith, 1234 Oak St., London, GB, 999990"; 

要 $address[] = "Wang Xiaoer, A28 Shangdi Sanjie St., Beijing, PRC, 100085"; 
10 $address[] = "Liu Laipin, 1345 Yiyuan St., Wuhan, PRC, 430001"; 
寺村 

2 /* 第 三 步 ， 我 们 需要 使 用 for 循环 来 将 数组 中 存放 的 数据 

3 写 入 到 CSV 文件 中 */ 

14 二 EN 人 = "07 SL « 92 SELEYT 

村 fwrite ($fh, S$address[$i]."\n"); 

16 } 

于 了 

18 // 最 后 ， 关 闭 打开 的 文件 

19 fclose ($fh); 

区 


运行 了 上 面 这 段 脚本 后 , 当前 目录 下 生成 了 一 个 名 为 fleCSV.csv 的 文件 ,打开 该 文件 ， 
你 会 发 现 $address 数组 中 的 数据 被 安安 稳 稳 地 存放 到 了 文件 中 。 

现在 ， 我 们 就 来 从 这 个 CSV 文件 中 读 取 数据 。 如 果 还 是 使 用 fgets0 函 数 的 话 ， 那么 就 
只 能 一 行 一 行 地 读 了 。 因 此 CSV 文件 和 普通 的 文本 文件 相 比 而 言 的 优势 就 不 复 存在 了 。 为 
了 解决 这 个 问题 ，PHP 提供 了 一 个 新 的 函数 : fgetcsvO。 这 个 函数 有 两 个 必要 参数 ， 一 个 
为 文件 名 ， 另 一 个 为 读 取 的 最 大 字符 数 。 

【 例 10.7】 从 CSV 文件 中 读 取 数据 。 

上 <?php 
名 // 检查 fileCSsV.csv 文件 是 否 存在 
3 if (file exists("fileCSV.csv")){ 
4 $fh = fopen("fileCSV.csv", "r"); 
可 S$info = fgetcsv($fh, 1000); 
6 foreach ($info as $key => $value) { 
学 echo "\$info[".$key."] ".$value."<br>"; 
8 } 

9 fclose ($fh); 

10 } 

bh 

在 例 10.7 的 这 段 脚本 中 ， 首 先 判断 了 fleCSV.csv 文件 是 否 存 在 于 当前 文件 夹 下 。 如 
果 存 在 , 则 以 只 读 的 方式 打开 该 文件 , 然后 用 fgetcsv0 函 数 将 文件 的 前 1000 个 字符 读 取 到 
$info 数组 中 。 需 要 注意 的 是 fgetcsv0 函 数 返 回 的 是 一 个 数组 。 然 后 用 foreach0 函 数 遍 历 了 
数组 Sinfo 的 值 。 


运行 这 段 脚本 后 ， 得 到 的 结果 如 下 : 
$info[0] John Smith 


sinfolll 1234 Oak St 
$info[2] London 
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$info[3] GB 
$info[4] 999990 


原来 ，fgetcsv0 函 数 只 读 取 了 第 一 行 的 内 容 ， 并 且 将 用 逗号 分 阳 开 的 值 存放 到 了 $info 
数组 的 每 个 元 素 当 中 。 如 果 我 们 想 遍 历 该 文件 中 的 所 有 内 容 , 就 得 将 上 面 的 这 段 脚 本 改 成 : 


. <?php 

2 // 检 查 filecSV.csv 文件 是 否 存 在 

3 if(file exists("fileCSV.csv")){ 

$fh = fopen("fileCSV.csv", "“r"); 

5 

6 //Read the data saved in the CSV file. 
i $fh = fopen("fileCSV.csv", "“r"); 

8 

9 Si = 0; 

10 while (!feof($fh)) { 

bh $info = fgetcsv($fh, 1000); 
if($info <> FALSE) { 

3 foreach ($info as $key => $value) { 
14 echo "[$i] [$key] $value<br>"; 
二 1 

16 SU 

Ty } else { 

18 die ("Reached the end of the file."); 
于 向 

20 水 

21 fclose ($fh); 

22 } 

> 


上 面 这 段 脚本 还 有 一 点 需要 说 明 ， 那 就 是 while 循环 中 那个 站 语句 的 条 件 。 为 什么 要 
有 这 么 一 句 呢 ? 因为 文本 文件 中 很 有 可 能 会 存在 空 行 ， 如 果 一 旦 碰 到 空 行 的 话 ， 那 么 我 们 
使 用 fgetcsv0 函 数 得 到 的 就 不 是 个 数组 了 ， 因 而 也 就 不 能 使 用 foreach 来 遍历 了 。 如 果 强 行 
使 用 foreach 来 遍历 不 是 数组 的 元 素 ， 则 得 到 了 就 只 能 是 系统 的 和 警告。 因此, 在 遍历 数组 之 
前 ,一 定 要 保证 遍历 的 对 象 是 个 数组 ， 而 “Sinfo <> FALSE” 就 是 用 来 保证 这 一 点 的 。 

最 后 来 看 看 如 何 读 取 TSV 格式 的 文本 文件 吧 。 

其 实 ，fgetcsvO 函 数 不 光 能 读 取 CSV 格式 的 文本 文件 ， 还 可 以 读 取 所 有 *SV 格式 的 文 
件 。 因 为 这 个 函数 的 第 三 个 参数 就 是 分 隔 符 ， 而 TSV 文件 里 的 值 都 是 以 “\t” 来 分 隔 的 。 
在 键盘 上 它 有 一 个 对 应 的 键 ， 那 就 是 Tab 键 。 在 记事 本 里 ， 如 果 按 一 下 Tab 键 ， 光 标 就 会 
缩 进 一 段 距离 。 我 们 可 以 按照 制作 CSV 文件 的 方式 制作 一 个 TSV 文件 ， 只 需要 把 逗号 换 
成 “\t” 即 可 。 

之 所 以 要 使 用 TSV 文件 的 根本 原因 是 ， 存 储 在 文本 文件 的 数据 里 可 能 真 的 会 用 到 去 
号 。 如 果 一 旦 如 此 , 而 我 们 又 将 这 些 数据 存放 在 了 一 个 CSV 文件 里 ， 那 某 一 条 数据 可 能 就 


10.1.5 ”实战 练习 : 用 文本 文件 做 数据 源 的 留言 


【目标 】: 制作 一 个 以 文本 文件 为 数据 源 的 留言 本 。 
【效果 】: 制作 的 效果 如 图 10-2 所 示 。 
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图 10-2 以 文本 文件 为 数据 源 的 留言 本 


【要 求 】 

口 使 用 所 学 的 内 容 做 一 个 以 文本 文件 为 数据 源 的 留言 本 ; 

口 这 个 留言 本 可 以 记录 留言 人 的 名 字 、 留 言 的 时 间 和 留言 的 内 容 ; 
口 留言 需 按照 从 新 到 旧 的 顺序 显示 。 


【 源 代码 】 

ph <?php 

2 rt 

3 * 检查 是 否 有 用 户 提交 了 留言 

4 * 若是 ， 以 追加 的 方式 打开 一 个 名 为 guestBook .tsv 的 文件 

5 * 并 在 其 末尾 追加 用 户 提交 的 留言 。 

6 sf 

Ti if (isset($ REQUEST['append'])) { // 判 断 用 户 是 否 提交 了 留言 

8 $fh append = fopen("guestBook.tsv", "a"); 

9 $comment = $ POST['userName'] ."\t". date ("Y-m-d") . "NE":$. POST 
LeomenEle Nn 

10 EE (OE append $comment); 

中 有 fclose ($fh append) ; 

BS } 

1 

14 人 

LS * 1。 ” 若 名 为 guestBook.tsrv 的 文件 存在 ， 打 开 该 文件 ， 

16 * ”否则 创建 一 个 变量 $message 用 来 存储 告知 用 户 

a * ”该 文件 不 存在 的 消息 。 

18 * 2. 用 fgetcsv() 函数 分 行 读 取 TSV 文件 中 的 内 容 ， 并 

19 * ”把 它们 存 入 一 个 名 为 $comment 的 数组 。 

20 * 3。. 创建 三 个 数组 ， 分 别 将 其 命名 为 $commentUser， 

2 * ”$commentDate， 和 $commentCon 分 别 对 应 一 个 用 

22 * ， 户 的 名 字 、 用 户 留言 的 日 期 和 用 户 的 留言 。 

23 * 4. 变量 $i 用 来 存储 TSV 文件 中 数据 的 行 数 。 每 读 一 行 

24 * ”变量 $i 的 值 加 1。 

25 高 扩 

26 $i = 0; 

号 if (@$fh read = fopen("guestBook.tsv", "r")) { 

28 while (!feof($fh read)) { 

29 $comment = fgetcsv($fh read, 1000, “\t"); 

30 if ($comment <> FALSE) { 

eh $commentUser[$i] = $comment [0]; 


// 三 个 数组 分 别 存储 用 户 的 姓名 
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$commentDate[$i] = $comment[1];// 用 户 留言 的 日 期 


$commentCont[$i] = $comment[2];// 用 户 留言 的 内 容 
Si++7 // 每 读 取 一 行 的 记录 , 变量 $i 加 1 
} 
} 
} else { 
$message = "The guestbook does not exist."; 


} 
> 
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN" 
"http:/V/www.w3.org/TR/htm14/strict.dtd"> 
<meta content="text/html; charset=UTF-8" http-equiv="Content-Type"> 
<html> 
<head> 
<title>Guest Book</title> 
<style> 
#myForm { 
width: 50%; 
margin: 0 15%; 
} 


</style> 
</head> 
<body> 


<br> 
<center> 
<?php 
/* 
* 1. 若 变 量 $i 大 于 0， 则 TSV 文件 中 有 数据 ， 反 之 
* ” 则 打印 变量 $message， 告 知 用 户 该 文件 不 存在 
* 2. 用 for 循环 来 凯 历 上 面 创 建 的 三 个 数组 ， 并 按照 
* ”从 新 到 旧 的 顺序 罗列 它们 
/ 
if ($i > 0) { // 若 变量 $i 为 零 ， 输 出 变量 $message 的 值 
for ($j=$i-1; $j>=0; $j--) { 
// 以 从 新 到 旧 的 方式 显示 留言 记录 
2> 
<table border="0" width="400"> 
< 
<td align="left" width="30%"> 
// 输 出 当前 指针 所 指 留言 记录 对 应 的 用 户 名 
<strong><?php echo $commentUser[$j]; ?> 
</strong> 
</td> 
<td align="center" width="40%"> 
// 输 出 当前 留言 记录 对 应 的 留言 日 期 
<strong><?php echo $commentDate[$j]; ?> 
</strong> 
</td> 
<td align="right" width="30 委 "> 
// 输 出 当前 留言 记录 对 应 的 楼 层 数 
<strong><?php echo "#". ($j+1); ?></ Strong> 
</td> 
</tr> 
2 
<td colspan="3" align="left"> 
<p></p> // 输 出 当前 留言 记录 对 应 的 留言 内 容 
<p><?php echo $commentCont[$j]; ?></p> 
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85 </td> 

86 </Er> 

87 <tr><td colspan="3" align="center"></td></tr> 

88 </table> 

89 <?php 

90 } 

91 } else { 

92 echo $message; 

93 

94 > 

95 </center> 

96 <hzr /> 

97 <div id="myForm"> 

98 <?php 

99 Ke 

100 * 关于 这 个 表单 ， 我 们 在 action 字段 里 指 定 的 文件 名 后 

后 * 加 上 了 一 个 问号 和 一 个 字符 串 ， 用 来 判断 用 户 是 否 提 

102 * 交 了 留言 

103 */ 

104 2 

105 <form action="Ex10-08.php?append" method="post"> 

106 <table border="0" width="600 px"> 

107 <tr> 

108 <td width="60%" align="left"> 

109 <label>Name</label> 

110 <input type="text" name="userName" /> 

1 </td> 

112 </tr> 

了 <tr> 

114 <td align="left"> 

115 <label>Comment</label> 

116 </td> 

17 <HEr> 

118 <tr> 

119 <td align="left"> 

120 <textarea name="Comment" cols="60"> 

2 Leave your comment here!</textarea> 

主人 人 </td> 

123 </tr> 

124 <tr> 

125 <td align="center"> 

126 <input type="submit" name="Submit" 
value="Submit" /> 

127 </td> 

128 </Er> 

129 </table> 

130 </form> 

3 </div> 

Ey </body> 

133 </html> 


10.2 ”使 用 XML 和 存 取 数据 


上 一 节 学 习 了 在 文本 文件 中 存储 数据 和 从 文本 文件 中 读 取 数据 。 在 本 节 里 ， 我 们 来 看 
看 如 何在 XML 中 存储 数据 和 从 XML 文件 中 读 取 数据 。 如 果 对 XML 不 太 了 解 ， 或 者 不 想 
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使 用 XML 来 存储 和 读 取 数据 的 话 ， 可 以 跳 过 本 节 的 内 容 ， 直 接 阅读 10.3 节 ， 使 用 数据 库 
存 取 数 据 。 

在 学 习 如 何 使 用 XML 存 取 数据 之 前 ， 先 来 看 看 什么 是 XML 。 

XML， 全 名 叫 可 扩展 标记 语言 (Extensible Markup Language) 。 按 照 W3C 的 说 法 ， 
XML 是 一 种 脱 生 于 SGML 的 非常 简单 、 灵 活 的 格式 。 最 初 ，XML 被 设计 来 解决 大 型 电子 
出 版 业务 当中 遇 到 的 挑战 。 而 现在 , XML 在 种 类 繁多 的 应 用 程序 间 进 行 数据 交换 当中 发 挥 
的 作用 也 日 渐 重 要 。 

XML 有 如 下 几 个 特点 : 

口 它 是 纯 描 述 类 的 标记 语言 。 不 带 任何 样式 标记 ， 只 是 用 来 以 结构 化 的 方式 存储 和 

传输 信息 。 

口 它 是 以 纯 文本 的 形式 存储 的 。 有 能 力 处 理 纯 文本 的 软件 都 可 以 处 理 XML 。 

口 XML 中 的 标签 是 用 户 自己 定义 的 。 标 签 之 间 的 关系 可 以 通过 它们 之 间 的 层级 结构 

得 到 展现 。 

如 果 读 者 对 XML 感 兴趣 ， 也 可 以 自行 搜索 相关 的 内 容 与 教程 。 从 现在 开始 ， 我 们 假 
设 读者 已 经 了 解 了 XML 是 什么 以 及 如 何 编写 XML 文件 。 在 此 基础 上 ， 我 们 应 该 要 使 用 
PHP 来 操作 XML， 即 构造 XML 从 XML 文件 中 读 取 数据 ， 向 XML 文件 中 写 入 数据 等 。 


10.2.1 加 载 和 读 取 XML 数据 


我 们 可 以 从 XML 格式 的 字符 串 中 读 取 XML， 也 可 以 从 一 个 XML 文件 中 读 取 XML 。 
通常 的 做 法 是 将 XML 存放 在 一 个 XML 文件 中 ， 然 后 用 simplexml load_file() 方 法 将 其 读 
取 到 simpleXMLElement 对 象 中 。 

在 例 10.9 中 ， 先 创建 一 个 XML 文件 ， 将 命名 为 notes 并 保存 在 当前 目录 下 。 其 内 容 
如 下 : 


下 <?xml version="1.0" encoding="UTF-8" ?> 

2 <notes> 

3 <note id="1"> 

4 <from>Jimmy Carter</from> 

5 <to>Routney Dickinson</to> 

6 <date>2013-03-12</date> 

学 <subject>Hello, Routeney!</subject> 

8 <noteBody> 

9 Hi! Old buddy! It's about five years since we last met each other! 

10 I'11 have a trip to Boston tomorrow, and would like to meet 
you there! 

FE Let me know if you are free then. 

32 </noteBody> 

.3 </note> 

14 </notes> 


然后 ， 可 以 使 用 下 面 的 语句 将 这 个 文件 的 内 容 读 取 到 一 个 变量 中 。 
【 例 10.9】 从 XML 文件 中 读 取 数据 。 


(i <?php 

2 

3 Vs 

4 * 在 这 里 我 们 可 以 用 var_dump () 或 print r() 函 数 
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5 * 来 检验 $xml0Object 是 否 是 一 个 simpleXMLElement 对 象 
6 */ 

$xmlObject = simplexml load file("notes.xml"); 
8 


9 foreach ($xmlObject->children() as $block) { 

10 

I // 获取 子 节点 的 文本 和 属性 

下 之 echo “<strong>".$block->from."</strong> send <strong>" 
| .$block->to."</strong> a note IDed <strong>".$block['id']. 
14 "</strong> on <strong>".$block->date."</strong>.<br>"; 
5 BCho "<hr /> 

16 

17 // 像 遍 历数 组 一 样 遍历 一 个 节点 

18 foreach ($block->children() as $tag => $text) { 

19 if ($tag <> "noteBody") { 

20 echo "<strong>".$tag."</strong>: ".$text; 

El echo "<br>"; 

22 } else { 

2 echo "<p>" .Stext7 

24 } 

之 5 1 

26 echo "<br><br>"7 

27 } 

28. 2> 


上 面 这 段 脚本 首先 使 用 simplexml load file0 函 数 加 载 了 notes.xml 文件 ， 并 将 其 赋值 
给 了 变量 $xmlObject。 按 照 之 前 的 说 法 ， 变 量 SxmlObject 是 个 simpleXMLElement 对 象 。 然 
后 用 foreach 语句 裔 历 这 个 对 象 。 在 遍历 对 象 时 , 使 用 了 simpleXMLElement 类 的 children() 
方法 来 获取 对 象 SxmlObject (也 就 是 note.xml 的 根 节点 ) 的 子 节点 。 这 个 children() 方 法 在 
遍历 某 节点 下 的 所 有 子 节点 是 非常 有 用 的 。 

除 此 之 外 ， 还 可 以 使 用 元 素 选择 器 “->” 来 获取 存储 在 某 个 节点 的 下 文本 。 在 上 面 这 
段 脚本 的 第 12 一 14 行 ， 变量 $block 对 应 的 是 该 XML 文件 中 的 “<note>” 节 点 。 我 们 分 别 
使 用 了 “S$block->to”、“Sblock->from” 和 “S$block->date ”获取 了 收 件 人 、 发 件 人 和 日 
期 三 个 节点 下 存储 的 文本 人 信息。 同时， 我们 还 使 用 “Sblock["id'] ”对 应 的 “<note>” 节 点 
下 的 属性 “id” 的 值 。 

这 段 脚本 输出 的 内 容 如 图 10-3 所 示 。 


[em mem 


| 大 wepywwwsres 


| 
2 ® | CE C7 


| 
| 帘 Favorites Ehttp//www greatwalltea/chapterl0/Ed0-09.php 租 " 国 -名 Page” safey- Toosv 属 - 


Jimmy Carter send Routney Dickinson a note IDed 1 on 2013-03-12. 


from: Jimmy Carter 

to: Routney Dickinson 
date: 2013-03-12 
subject: Hello, Routeney! 


Hi! 01d buddy! It’s about five years since we last met each other! I’1] have a trip to Boston 
tomorrow, and would like to meet you there! Let me know if you are free then. 


图 10-3 ”加载 和 读 取 XML 数据 


例 10.9 中 使 用 的 notes.xml 文件 其 实 也 可 以 用 字符 串 的 形式 存储 到 一 个 变量 中 ， 就 像 
下 面 这 样 : 
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1 $xmlString = <<<XML 

加 <?xml version="1.0" encoding="UTF-8" ?> 

3 <notes> 

4 <note id="1"> 

3 <from>Jimmy Carter</from> 

6 <to>Routney Dickinson</to> 

沉 <date>2013-03-12</date> 

8 <subject>Hello, Routeney!'</subject> 

9 <noteBody> 

10 Hi! Old buddy! It's about five years since we last met each other! 

IL I'11 have a trip to Boston tomorrow, and would like to meet 
you there! 

2 Let me know if you are free then. 

3 </noteBody> 

14 </note> 

15 </notes> 

16 XML; 


然后 使 用 simplexml load_string() 方 法 读 取 这 个 字符 串 ， 其 他 的 所 有 操作 方法 都 是 一 样 
的 。 需 要 说 明 的 是 ，simplexml load_string0 和 simplexml load file() 两 个 函数 返回 的 都 是 
simpleXMLElement 对 象 。 


10.2.2 ”修改 XML 文件 中 的 数据 


在 学 习 了 如 何 加 载 和 读 取 XML 文件 中 的 数据 之 后 , 再 来 看 看 如 何 修改 XML 文件 中 的 
数据 。 

【 例 10.10】 修改 XML 文件 中 的 数据 。 

下 <?php 


2 // 读 取 文 件 notes .xml 中 的 XML 数据 

3 $xmlObject = simplexml load file("notes.xml"); 
4 

5 // 设 置 节点 “to” 的 值 

6 $xmlObject->note->to = 'Anthony Tsu'; 
沈 

8 // 打 印 修改 后 XML 数据 

9 $xmlString = $xmlObject->asXML(); 

10 echo $xmlSstring; 

入 于 

2 // 以 只 写 方式 打开 notes .xml 文件 

1 3 // 并 用 $xmlString 的 值 获 盖 原 有 数据 

14 $fh = fopen("notes.xml", "w"); 
fwrite($fh, $xmlString); 

16 fclose ($fh); 

证 着 cz 


运行 了 上 面 这 段 脚本 后 ， 我 们 再 运行 例 10.9 中 的 那 段 脚 本 ， 就 会 发 现 to 节点 的 值 真 
的 发 生 了 如 图 10-4 所 示 的 变化 。 

在 例 10.10 这 段 脚 本 里 ， 我 们 使 用 simplexml load file0 读 取 到 变量 $SxmlObject 中 。 这 
时 ， 变 量 $SxmlObject 是 一 个 simpleXMLElement 对 象 。 紧 接着 ， 我 们 使 用 元 素 选择 符 “->” 
选择 “to” 节 点 ， 并 将 它 的 值 修 改 成 了 “Anthony Tsu”。 然 后 ， 使 用 simpleXMLElement 
对 象 的 asaXMLQO 也 可 以 写成 saveXMLO) 方法 将 这 个 修改 后 的 simpleXMLElement 对 象 
转换 成 一 个 字符 串 并 将 这 个 字符 串 赋 值 给 了 变量 $xmlString。 最 后 ， 我 们 使 用 了 fopen()、 
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fwrite() 和 felose() 方 法 将 变量 $xmlString 的 值 以 覆盖 的 方式 保存 到 了 notes.xml 中 。 


ap /fom reat 


9 rE 


| 将 Favorites 攻 http//www.greatwalltea/chapterl0/Ed0-09.php 丛 "~ 国 ~ 本 车 Page safey” Tooksv 各 - 


| Jimmy Carter send[NNEEGSN a note IDed 1 on 2013-03-12. 
| from: Ji Carter 


date: 2013-03-12 
subject: Hello, Routeney! 


Hi! Old buddy! It’s about five years since we last met each other! I’1] have a trip to Boston 
tomorrow, and would like to meet you there! Let me know if you are free then. 


图 10-4 修改 XML 中 的 数据 


于 是 ，“to” 节 点 的 值 由 原来 的 “Routney Dickinson” 变 成 了 现在 的 “Anthony Tsu”。 
10.2.3 向 XML 文件 中 添加 数据 


现在 ， 我 们 再 试 着 向 notes.xml 中 添加 一 条 记录 ， 内 容 如 表 10-2 所 示 : 
表 10-2 文件 notes.xml 中 的 一 条 新 记录 


From Samuel Johnson 

To Annie Hathaway 

Date 2013-3-17 

Subject Reminder 

Body Don't forget the meeting at 2 p.m. this aftermoon. 


从 表 10-2 所 示 中 列 出 的 数据 来 看 ，notes.xml 文件 中 的 第 二 条 记录 是 一 个 名 叫 Samuel 
Johnson 的 人 向 另 一 个 名 叫 Annie Hathaway 的 人 发 出 的 提醒 消息 。 消 息 的 内 容 是 让 Annie 
不 要 忘记 参加 今天 下 午 两 点 的 会 议 。 

为 了 将 这 批 数据 存储 到 notesxml 文件 中 ， 例 10.11 展示 的 脚本 使 用 到 
simpleXMLElement 对 象 的 addAttribute()、addChild 和 count() 方 法 。 其 中 ，count() 方 法 返回 
的 是 XML 数据 中 当前 节点 拥有 的 一 级 子 节点 数 ; 而 addAttribute0 和 addChild0) 都 拥有 两 个 
参数 : 第 一 个 参数 为 节点 标签 ， 第 二 个 参数 为 节点 值 。 


【 例 10.11】 向 XML 文件 中 添加 数据 。 

1 <?php 

之 // 读 取 文件 notes .xml 中 的 XML 数据 

3 $xmlObject = simplexml]l load file("notes.xml"); 
4 

忆 // 计 算 当 前 存储 的 note 记录 的 ID 

6 $count = $xmlObject->count() + 1; 

没 

8 // 在 根 节点 notes 下 添加 一 个 note 节点 ， 并 将 其 赋值 给 变量 Snote 
9 Snote = $xmlObject->addChild("note"); 

10 

11 // 为 note 节点 添加 属性 “id”, 并 将 变量 $count 的 值 赋 给 它 
下 之 $note->addAttribute('id', $count); 

bh 
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全 // 为 note 节点 添加 “to” 等 五 个 子 节点 


5 $note->addChild('from', "Samuel Johnson') 

16 $note->addChild('to', "Rnnie Hathaway'); 

$note->addChild('date', "2012-03-177) 7 

18 $note->addChild('subject', "Reminder') 

19 $note->addChild('noteBody', ‘Don\'t forget the meeting at 2 p.m. 
this afternoon.'); 

20 

21 // 将 修改 后 的 xML 数据 以 字符 串 的 形式 赋值 给 变量 $xmlString 

公公 $xmlString = $xmlObject->asXML(); 

2 

24 // 将 变量 $xmlstring 的 值 以 覆盖 的 方式 存储 到 文件 notes .xml 中 

FL $fh = fopen("notes.xml", "w"); 

26 fwrite ($fh, $xmlString); 

27 fclose ($fh); 

28 

29 // 引 入 例 10. 9 中 的 脚本 

30 include('Ex10-09.php'); 

SU > 


该 脚本 。 否 则 这 一 条 数据 会 以 不 同 的 ID 重复 写 入 notes.xml 文件 


注意 ， 不 要 重复 运行 
行 的 结果 如 图 10-5 所 示 。 


中 。 上 面 这 段 脚本 运 


Ds 
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Jimmy Carter scnd Anthony Tsu a notc IDcd 1 on 2013-03-12. 


from: Jimmy Carter 

to: Anthony Tsu 

date; 2013-03-12 
subject: Hello, Routeney! 


Hi! 01d buddy! It’s about five years since we last met each other! I’1] have a trip to Boston 
tomorrow, and would like to meet you there! Let me know if you are free then. 


Samuel Johnson send Annie Hathaway a note IDed 2 on 2012-03-17. 


from: Samuel Johnson 
to: Annie Hathaway 
date: 2012-03-17 
subject: Reminder 


Don' t forget the meeting at 2 p.m this afternoon. 


图 10-5 向 XML 文件 中 添加 数据 


例 10.11 的 这 段 脚本 ,使 用 了 SimpleXMLElement 对 象 的 addAttribute 和 addChild 方法 。 
我 们 知道 $xmlObject 变量 中 存储 的 是 从 notes.xml 文件 中 读 取 的 所 有 XML 数据 。 也 就 是 说 
当前 这 个 SimpleXMLElement 对 象 的 指针 指向 的 是 根 节点 ， 也 就 是 “notes” 节 点 。 当 在 该 
节点 下 执行 addChild() 方 法 时 , 就 会 在 该 节点 下 添加 一 个 指定 的 子 节点 。 简 而 言 之 ,addChild0 
方法 可 以 帮助 我 们 在 当前 节点 下 创建 一 个 指定 的 子 节点 。 而 addAttribute0 也 是 类 似 的 ， 可 
以 在 当前 节点 下 为 其 创建 一 个 指定 的 属性 。 


10.2.4 遍历 XML 文件 中 的 数据 


之 前 提 到 的 simplexml_load_file0 函 数 , 其 实 它 的 第 二 个 参数 就 是 指定 将 文件 中 的 内 容 
读 取 到 什么 类 型 的 对 象 中 ， 这 个 参数 的 默认 值 是 SimpleXMLElement 对 象 。 如 果 我 们 将 其 
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指定 为 SimpleXMLIterator， 那 么 就 可 以 使 用 SimpleXMLIterator 对 象 提供 的 遍历 数据 的 方 
法 ， 轻 松 方便 地 在 XML 各 元 素 间 游 走 了 。 


来 看 一 个 例子 。 
【 例 10.12】 遍历 XML 文件 中 的 数据 。 

Ls <?php 

2 $xmlObject = simplexml load file('notes.xml', 'SimpleXMLIterator'); 
总 $xmlObject->rewind(); 

5 $i=0; 

6 while ($i<$xmlObject->count ()){ 

吕 foreach ($xmlObject->current() as $key => $value) { 

8 echo $key.': '.$value.'<p>"; 

这 } 

10 $xmlObject->next (); 

El $7 

42 } 

业 3 2> 


在 这 段 脚 本 中 ， 我 们 使 用 simplexml load file0 函 数 的 SimpleXMLIterator 参数 把 
notes.xml 文件 中 的 内 容 读 取出 来 赋值 给 了 变量 $xmlObject。 这 时 变量 $xmlObject 就 是 一 个 
SimpleXMLIterator 对 象 。 SimpleXMLIterator 类 是 由 SimpleXMLElement 扩展 而 来 , 继承 了 
SimpleXMLElement 类 的 所 有 属性 和 方法 。 这 也 是 为 什么 我 们 可 以 在 脚本 的 第 6 行使 用 
count() 方 法 的 原因 。 

在 上 面 这 段 脚 本 中 , 除了 count() 方 法 之 外 , 其 他 的 方法 都 是 SimpleIterator 特有 的 方法 。 
其 中 ，rewind0 方 法 用 于 移动 当前 指针 到 第 一 个 元 素 。 注 意 ， 我 们 只 有 执行 了 
SimpleXMLIterator 对 象 的 rewind() 方 法 后 ， 执 行 其 他 的 方法 才 会 有 结果 ， 和 否则 得 到 的 结果 

- 律 为 “NULL”。 接 下 来 ，current() 方 法 返回 当前 节点 的 内 容 ，next(0 方 法 将 指针 移动 到 

下 一 节点 。 除 此 之 外 ，SimpleXMLElement 对 象 还 有 诸如 getChildren() 这 个 用 来 获取 当前 结 

点 下 所 有 子 节点 的 方法 、hasChildren() 这 样 判断 当前 节点 下 是 否 拥有 子 节点 的 方法 ， 以 及 
valid0 这 样 用 来 判断 当前 节点 是 否 为 一 个 合法 节点 的 方法 。 

关于 这 些 方法 的 具体 内 容 , 读 者 可 以 参考 PHP 手册 中 关于 操作 XML 数据 的 相关 内 容 。 
可 以 访问 如 下 网 址 : 


http://www.php .net/manual/en/book.simplexml.php。 


SimpleXMLIterator 类 通常 是 在 对 xPath 语法 不 太 了 解 的 情况 下 ， 需 要 操作 XML 数据 
时 使 用 的 。 如 果 你 对 xPath 有 所 了 解 ， 也 可 以 尝试 用 xPath 来 操作 XML 数据 。 这 时 ， 就 没 
有 必要 使 用 SimpleXMLIterator 对 象 了 。 因 为 SimpleXMLElement 类 为 我 们 提供 了 使 用 xpath 
定位 XML 元 素 的 方法 。 


来 看 一 个 例子 。 

【 例 10.13】 SimpleXMLElement 类 的 xpath( 方 法 。 

五 <?php 

2 $xmlObject = simplexml load file('notes.xml'); 

3 

4 // Obtains the children of the "note' node 

三 Snotes = $xmlObject->xpath('//note'); 

6 

这 //Note that the return value of the xpath() method is an array 


= 这 
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8 foreach ($notes as Snote) { 

9 foreach ($note as $key => $value) { 
10 echo $key.': '.$value.'<p>"; 

11 1y 

2 全 

3 


在 上 面 这 段 脚本 中 ， 我们 使 用 了 simple load file0 函 数 的 默认 参数 将 notes.xml 文件 中 
的 内 容 赋 值 给 了 变量 $xmlObject。 这 时 变量 $xmlObject 是 一 个 SimpleXMLElement 对 象 。 
然后 我 们 使 用 了 该 对 象 的 xpath0 方 法 查找 名 为 “note” 的 节点 ， 并 将 所 有 “note” 节 点 的 
内 容 存 入 数组 Snotes 中 。 接 下 来 ,用 foreach 语句 遍历 了 数组 Snotes， 输 出 了 notes.xml 文件 
中 的 内 容 。 

值得 注意 的 是 xpathO 对 象 中 使 用 的 xpath 表达 式 “/note”。 这 个 表达 式 中 使 用 的 “/” 
指 的 是 在 所 有 节点 中 查找 其 后 跟随 的 节点 。 这 里 “//note” 意 思 就 是 在 所 有 的 节点 中 查找 名 
为 “note” 的 节点 。 关 于 更 多 xPath 语法 的 内 容 ， 读 者 可 以 参考 w3school 提供 的 XPath 教 
程 。 可 以 访问 如 下 网 址 : 

http://www.w3cschool.cn/index-14.html 


例 10.12 和 例 10.13 两 段 脚本 虽然 使 用 了 不 同 的 方法 ， 但 输出 的 结果 却 是 如 出 一 略 。 
这 两 段 脚本 输出 的 结果 如 图 10-6 所 示 。 
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date; 2013-00-12 
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图 10-6 殊途同归 的 两 段 脚 本 


人 
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10.3 ”使 用 数据 库存 取 数 据 


之 前 掌握 了 如 何 使 用 文本 文件 和 XML 文件 来 存 取 数据 。 但 是 我 们 发 现 用 文本 文件 和 
XML 文件 来 存 取 数据 时 ， 有 一 点 不 太 方便 ， 那 就 是 删除 数据 。 如 果 我 们 需要 删除 数据 ， 就 
需要 把 所 有 的 数据 都 读 取出 来 放 入 若干 的 变量 或 者 数组 中 ， 剔 除 掉 与 需要 删除 的 数据 相关 
的 变量 或 数组 元 素 ， 再 把 剩余 的 数据 拼 在 一 起 以 覆盖 的 方式 写 入 原始 文件 中 。 操 作 起 来 十 
分 地 繁琐 复杂 。 对 于 大 量 的 数据 来 说 ， 资 源 消耗 将 呈 几 何 极 数 增长 。 

为 了 避免 出 现 这 种 情况 ， 可 以 使 用 关系 型 数据 库 来 存 取 数据 。 通 过 使 用 SQL 的 
SELECT、CREATE、UPDATE 和 DELETE 等 语句 对 数据 库 进 行 操作 ， 使 得 增加 、 修 改 和 
删除 指定 的 数据 变 得 非常 地 方便 。 

在 本 节 里 ， 所 有 的 内 容 都 将 围绕 一 个 简单 的 图 书 管理 系统 展开 介绍 。 我 们 可 以 使 用 这 
个 图 书 管理 系统 来 检查 图 书 的 存放 和 使 用 情况 以 及 读者 们 对 阅读 图 书后 产生 的 评价 。 这 个 
图 书 管理 系统 使 用 的 是 MySQL 数据 库 ， 架 构 如 图 10-7 所 示 。 


BooksID 
ReaderlD 
Date 
UseType 


FK1 
FK2 


图 10-7 图 书 管理 系统 数据 架构 图 


10.3.1 数据 库 基础 


为 了 看 懂 如 图 10-13 所 示 的 数据 架构 图 ， 我 们 需要 掌握 一 些 关于 数据 库 的 基础 知识 。 

一 个 关系 型 数据 库 是 由 表 以 及 表 和 表 之 间 的 关系 组 合 起 来 的 ， 这 就 是 为 什么 这 种 类 型 
的 数据 库 称 为 关系 型 数据 库 。 在 我 们 这 个 数据 库 里 一 共有 五 张 表 ， 它 们 分 别 是 : Books 表 、 
Reader 表 、UserLog 表 、Comment 表 和 BookCategory 表 。 表 10-3 所 示 描 述 了 这 些 数据 库 
表 记 录 的 内 容 : 
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表 10-3 ”图书 管理 系统 中 用 到 的 数据 库 表 


数据 库 表 描 述 
Books 于 存放 诸如 书 名 、 作 者 、 出 版 日 期 等 与 图 书 有 关 的 信息 
Reader 于 存放 诸如 姓名 、 性 别 、 出 生年 月 、 方式 等 与 读者 有 关 的 信息 
于 存放 诸如 使 用 类 本 借阅 时 限 等 与 图 书 借阅 行为 有 关 的 信息 ， 该 表 与 Books 和 
” Reader 两 张 表 之 间 存 在 着 引用 关系 
于 存放 读者 在 借阅 的 图 书后 产生 的 评价 , 该 表 与 Books 和 Reader 两 张 表 之 
me 间 存 在 着 引用 关系 
BookCategory 于 给 存放 在 Books 表 中 的 图 书 进行 分 类 ， 该 表 与 Books 表 之 间 存 在 着 引用 关系 
通过 表 10-3 所 示 ， 我 们 大 致 了 解 了 这 五 张 表 的 用 途 。 现 在 就 来 看 看 如 何在 MySQL 数 


居 库 服务 器 上 创建 这 些 数据 库 表 吧 。 

如 果 你 和 我 一 样 使 用 的 是 安装 有 Windows 7 操作 系统 的 计算 机 ， 并 且 将 MySQL 服务 
器 软件 安装 在 C 盘 ， 那 么 你 可 以 在 C 盘 下 找到 MySQL 的 安装 文件 夹 ， 记 录 该 文件 夹 的 路 
径 。 我 的 路 径 是 : 

C:\Program Files\EasyPHP-12.1\mysql\ 

选择 “开始 ”|“ 附 件 ”| “命令 提示 符 ” 命 令 ， 打 开 命 令 窗 口 。 然 后 在 当前 路 径 下 运行 
如 下 的 命令 : 

cd \"Program Files"\EasyPHP-12.1\mysql\bin 

进入 MySQL 安装 路 径 下 的 bin 文件 夹 。 随 后 运行 如 下 命令 ， 以 使 用 root 用 户 登 录 
MySQL 服务 器 : 

mysql -uroot 

为 了 演示 的 方便 ， 我 们 数据 库 root 用 户 并 没有 加 密 。 如 果 你 在 安装 MySQL 服务 器 的 
过 程 中 为 root 用 户 指 定 了 密码 ， 那 么 请 在 上 面 的 命令 后 添加 “-p” 人 参数， 并 写 上 密码 。 


图 10-8 ”MySQL 命令 行 窗 口 


如 果 看 到 了 如 图 10-8 中 所 示 的 MySQL 欢迎 信息 ， 那 么 恭喜 你 成 功 进 入 了 MySQL 命 
令 行 窗口 。 如 果 看 到 了 如 下 所 示 的 错误 提示 : 


sls 
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ERROR 2003 (HY000): Can't connect to MYSQL server on "localhost' (10061) 

多 半 是 由 于 MySQL 数据 库 服务 没有 启动 。 只 需要 启动 你 的 MySQL 数据 库 服务 即 可 。 

如 果 成 功 启 动 了 MySQL 数据 库 服务 ， 并 且 进 入 了 MySQL 命令 行 窗口 ， 那 就 着 手 为 
这 个 图 书 管理 系统 创建 数据 库 吧 。 

(1) 创建 数据 库 

在 光标 闪 动 的 “mysql> ”命令 提示 符 后 输入 如 下 命令 : 


CREATE DATABASE LibraryManager; 


按 下 回 车 键 后 ， 系 统 返 回 如 下 信息 : 

Query OK, 1 row affected (0.02sec) 

表明 ， 一 个 名 为 LibraryManager 的 数据 库 创建 成 功 。 

你 可 以 使 用 如 下 的 命令 来 查看 当前 MySQL 服务 器 上 的 所 有 数据 库 : 

SHOW DATABASES; 

如 果 你 可 以 在 这 条 命令 的 回 显 信息 中 ， 找 到 我 们 刚才 创建 的 数据 库 ， 那 么 数据 库 
LibraryManager 就 创建 成 功 了 。 

(2) 创建 数据 库 表 

在 创建 数据 库 表 之 前 ， 需 要 使 用 如 下 的 命令 指定 一 个 数据 库 : 

USE LibraryManager; 

这 时 ， 系 统 回 显 “Database changed”， 表 明 数 据 库 切 换 成 功 。 也 就 是 说 ， 我 们 可 以 创 

这 里 只 演示 如 何 创建 数据 库 表 Books。 其 他 的 数据 库 表 大 家 可 以 重复 如 下 的 步骤 进行 


创建 。 
在 光标 内 动 的 “mysql>” 后 输入 如 下 的 命令 : 
CREATE TABLE books ( 
bookid int NOT NULL AUTO INCREMENT PRIMARY KEY, 
title CHAR(50) ， 
author CHAR(50), 
ISBN CHAR(50), 
publisher CHAR(50), 
pubdate DATE NOT NULL, 
catid INT NOT NULL 
) DEFAULT CHARACTER SET utf8 ENGINE=InnoDB; 


这 条 命令 中 使 用 了 CREATE TABLE 关键 字 ， 告 诉 MySQL 服务 器 我 们 要 为 当前 数据 
库 创建 一 张 名 为 Books 的 表 。 括 号 里 的 内 容 是 定义 数据 库 表 Books 的 结构 。 按 照 之 前 的 规 
划 , 数据 库 表 Books 有 七 列 , 它们 分 别 是 Bookid、Title、Author、 ISBN、Publisher、Pubdate 
和 Catid。 每 列 可 以 容纳 的 数据 类 型 不 同 , 在 定义 数据 库 表 时 需要 指定 , 比如 Bookid 和 Catid 
两 列 只 能 容纳 整数 ，Pubdate 列 只 能 容纳 时 间 ， 其 余 列 只 能 容纳 字符 串 。 在 命令 的 最 后 定义 
了 数据 表 中 存储 的 数据 使 用 的 字符 集 和 数据 引擎 。 在 定义 好 一 切 后 ， 按 下 回 车 键 。 当 系统 
回 显 : 


Query OK, 0 rows affected (0.19 sec) 
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表明 数据 库 表 Books 已 经 创建 成 功 。 这 时 , 可 以 使 用 DESCRIBE 命令 来 打印 数据 库 表 
Books 的 结构 ， 如 图 10-9 所 示 。 


图 10-9 数据 库 表 books 的 数据 结构 


在 命令 DESCRIBE books 命令 的 回 显 
刚才 定义 的 表格 每 一 列 的 情况 。 值 得 
其 值 会 根据 上 一 行 的 内 容 自 动 添加 。 

现在 ， 我 们 可 以 按照 之 前 的 规划 ， 把 其 他 四 张 表 添加 到 数据 库 中 。 添 加 完成 后 的 结果 
如 图 10-10 所 示 。 


信息 中 ， 我 们 看 到 了 一 张 表 。 十 分 详尽 地 罗列 了 
的 是 , 表格 的 第 一 列 为 该 表 的 主键 , 不 允许 为 空 ， 


画 Admnstrator Command Prompt mysql urool pe | 


图 10-10 ”图书 管理 


里 的 数据 表 
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(3) 向 数据 表 中 添加 数据 

在 添加 了 所 有 计划 的 数据 库 表 之 后 ， 我 们 就 可 以 向 数据 库 中 添加 数据 了 。 先 在 
bookcategory 表 中 添加 一 条 数据 : 

INSERT INTO bookcategory SET 

category="'Literature'; 

这 里 使 用 了 INSERT INTO 关键 字 用 来 向 bookcategory 表 中 添加 数据 。 之 后 使 用 SET 
关键 字 来 设置 bookcategory 表 中 当前 行 与 category 列 〈 或 称 为 “字段 ”) 相交 的 单元 格 的 
值 为 “Literature”。 

这 条 命令 也 可 以 写成 如 下 的 样子 : 

INSERT INTO bookcategory 

(category) VALUES ('Computer Science'); 
如 果 需 要 同时 指定 多 个 字段 的 值 ， 可 以 用 逗号 将 它们 隔 开 ， 就 像 下 面 这 样 ; 


INSERT INTO tableName 
(fieldl, field2, ...) VALUES (valuel, value2, ...) 


(4) 查看 数据 表 中 存储 的 数据 
无 论 数据 表 中 是 否 存 在 数据 ， 我 们 都 可 以 使 用 SELECT 语句 来 查看 数据 表 的 内 容 。 只 
不 过 ， 如 果 数 据 表 中 没有 数据 ，SELECT 指令 返回 的 结果 是 : 


Empty set (0.00 sec) 


上 一 步 中 ， 我 们 在 bookcategory 表 中 添加 了 两 条 数据 。 现 在 ， 我 们 再 查询 一 下 
bookcategory 表 中 存储 的 数据 : 


SELECT * FROM bookcategory; 


这 条 指令 运行 的 结果 是 一 张 表格 ， 列 出 了 bookcategory 表 中 现 有 的 数据 。 其 中 ，“*” 
做 为 通配符 ， 代 表 所 有 的 列 。 运 行 该 指令 后 ， 一 张 两 行 两 列 的 表格 被 打印 了 出 来 。 其 中 第 
一 列 为 “catid”， 第 二 列 为 “category”。 在 上 一 步 中 ， 指 定 了 category 的 值 ， 并 没有 指定 
catid 的 值 。 但 是 “catid” 具 有 AUTO _INCREMENT 属性 ， 所 以 系统 会 在 上 一 条 记录 的 值 
的 基础 上 加 1 再 走 入 当前 记录 中 。 若 上 一 条 记录 不 存在 ， 则 当前 记录 的 值 为 1。 

我 们 也 可 以 通过 添加 WHERE 从 句 来 显示 需要 的 内 容 ， 比 如 : 


SELECT * FROM bookcategory WHERE catid=2; 


当 指 定 了 只 显示 catid 为 2 的 记录 时 ， 系 统 就 只 返回 了 数据 表 中 的 第 二 条 记录 。 
如 果 只 需要 显示 某 条 记录 的 某 一 列 的 值 时 ， 我 们 可 以 在 SELECT 关键 字 后 指定 该 列 ， 
就 像 下 面 这 样 : 


SELECT category FROM bookcategory WHERE catid=2; 


这 时 ， 显 示 的 表格 除 表 头 外 只 有 一 行 一 列 ， 也 就 是 cateid 为 2 的 那 条 记录 的 category 
字段 的 值 。 

(5) 修改 指定 记录 某 字段 的 值 

如 果 需 要 修改 某 条 记录 指定 字段 的 值 ， 我 们 可 以 使 用 UPDATE 命令 。UPDATE 命令 
的 格式 如 下 : 


.224 
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UPDATE tableName SET 
fieldl=valuel, 
field2=value2, 


比如 ， 我 们 想 要 把 bookcategory 表 中 的 第 二 条 记录 的 category 字段 的 值 从 “Computer 
Science” 修 改 为 “Science”， 就 可 以 使 用 下 面 这 条 命令 : 


UPDATE bookcategory SET category="'Science" 
WHERE catid=2; 


当 系 统 返 回 如 下 信息 时 : 


Query OK, 1 row affected (0.05 sec) 
Rows matched: 1 Changed:1 Warnings: 0 


指定 记录 的 category 字段 的 值 修改 成 功 。 这 时 再 使 用 SELECT 命令 显示 bookcategory 
表格 存储 的 所 有 数据 时 ， 就 会 发 现 新 的 数据 奉 代 了 原 有 的 数据 。 

(6) 删除 指定 的 记录 

如 果 需 要 删除 某 条 记录 ， 我 们 可 以 使 用 DELETE 命令 。DELETE 命令 的 格式 如 下 : 

DELETE FROM tableName WHERE conditions 

假如 ,我们 需要 删除 bookcategory 表 中 的 catid 为 2 的 记录 时 , 可 以 使 用 下 面 这 条 命令 : 

DELETE FROM bookcategory WHERE catid=2; 

当 系统 返回 如 下 信息 时 

Query OK, 1 row affected (0.00 sec) 

证 明 删 除 成 功 。 这 时 我 们 再 使 用 SELECT 命令 显示 bookcategory 表 中 的 数据 时 ， 就 看 
不 到 这 条 记录 了 。 


10.3.2 ”数据 表 之 间 的 关系 


在 数据 库 中 ,每 张 表 独立 存在 ,但 又 正如 图 10-7 所 示 互 相 有 着 联系 。 当 某 张 表格 中 的 
某 条 记录 的 某 一 个 字段 发 生变 化 时 , 相关 的 表格 中 存在 的 数据 也 会 发 生变 化 。 在 本 小 节 里 ， 
我 们 就 讨论 一 下 数据 库 中 数据 表 之 间 的 关系 。 
数据 表 之 间 存 在 的 关系 可 以 分 为 “一 对 一 ”和 “一 对 多 ”两 种 关系 ': 
口 所 谓 “ 一 对 一 ” 指 的 是 一 张 表 和 另 一 张 表 之 间 存 在 着 对 应 关系 ， 比 如 图 10-7 所 示 
中 Bookcategory 和 Books 两 张 表 之 间 的 关系 就 是 一 对 一 的 关系 , 其 中 Bookcategory 
表 的 主键 CatID 做 为 Books 表 的 外 键 让 这 两 张 表 联系 在 了 一 起 。 
口 所 谓 “ 一 对 多 ” 指 的 是 一 张 表 与 多 张 表 之 间 存 在 着 的 对 应 关系 ， 比 如 图 10-7 所 示 
中 Comment、Books 和 Reader 三 张 表 之 间 存 在 着 “一 对 多 ”的 关系 ， 其 中 Books 


按照 通常 的 看 法 ， 应 该 还 有 一 种 “多 对 多 ”的 数据 表 间 关系 。 假设 我 们 有 表 A 和 表 B 两 张 表 , 表 A 

中 的 一 条 记录 可 以 对 应 表 B 中 的 多 条 记录 ， 同 时 表 B 中 一 条 记录 也 可 能 对 应 表 A 中 的 多 条 记录 。 这 样 一 
来 ， 两 张 表 之 间 的 关系 就 是 “多 对 多 ”的 。 其 实 ， 我 们 可 以 把 这 个 多 对 多 的 关系 简化 成 两 个 一 对 多 的 
ws 
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表 和 Reader 表 的 主键 BooksID 和 ReaderID 做 为 Comment 表 的 外 键 将 它们 联系 在 
了 一 起 。 
这 里 说 到 主键 和 外 键 的 概念 ， 也 顺便 提 一 句 。 所 谓 主键 指 的 是 一 张 数据 表 中 用 来 区 分 
-条 记录 与 另 一 条 记录 的 字段 ,同一 张 表 中 的 每 个 记录 的 主键 值 都 必须 不 同 。 通 常情 况 下 ， 
我 们 会 使 用 “编号 ”来 做 一 张 数据 表 的 主键 。 所 谓 外 键 ， 就 是 指 一 张 数据 表 中 与 其 他 数据 
表 中 的 主键 相同 的 非 主键 字段 ， 它 是 用 来 联络 数据 表 之 间 的 关系 的 。 
在 本 节 里 ， 我 们 就 从 以 两 种 表 间 关系 为 例 来 讨论 一 下 数据 表 之 间 的 关系 。 


1. 一 对 一 的 表 间 关系 


在 讨论 一 对 一 的 表 间 关系 之 前 ， 先 在 Bookcategory 和 Books 两 张 表 中 添加 一 些 数据 。 
具体 请 参见 图 10-11 所 示 的 内 容 。 


图 10-11 一 对 一 的 表 间 关系 


现在 ，Books 表 中 有 4 条 记录 ， 而 Bookcategory 表 中 有 2 条 记录 。 两 张 表 之 间 存 在 着 
-对 一 的 关系 ， 因 为 Bookcategory 表 的 主键 CatID 是 Books 表 的 外 键 ， 而 Books 表 中 有 且 

仅 有 一 个 外 键 。 现 在 可 以 把 两 张 表 的 数据 结合 起 来 ， 输 出 我 们 想 要 输出 的 部 分 。 比 如 想 查 
找 Books 数据 表 中 由 Self 出 版 社 出 版 的 图 书 所 属 的 图 书 类 型 ,那么 就 可 以 执行 如 下 的 命令 ; 

SELECT books.title, books.author, bookcategory.category FROM books, bookcategory 

WHERE books.catid = bookcategory.catid AND 

books.publisher = 'Self'; 

再 比如 ， 我 们 想 要 查找 Books 数据 表 中 图 书 类 型 为 “Computer Science” 的 图 书 列表 ， 
那么 就 可 以 执行 如 下 的 命令 : 

SELECT books.title, books,author, bookcategory.category FROM books, bookcategory 

WHERE books.catid = bookcategory.catid AND 

books.catid = 2; 

在 这 两 条 命令 中 ， 我 们 使 用 SELECT 分 别 从 两 张 表 中 读 取 了 若干 字段 ， 然 后 使 用 
WHERE 从 名 指定 了 两 表 之 间 的 关系 和 查找 条 件 。 值 得 我 们 注意 的 是 : 

口 SELECT 关键 字 后 面 跟随 的 字段 名 称 前 面 都 带 上 了 表 名 ; 
口 指定 两 张 表 之 间 的 关系 时 ， 我 们 用 等 号 将 两 张 表 里 相同 的 字段 连接 在 了 一 起 ; 
口 AND 是 逻辑 运算 符 ， 表 示 只 有 当前 后 两 个 条 件 都 满足 时 ， 才 会 显示 结果 。 
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上 面 这 两 条 命令 的 结果 如 图 10-12 所 示 。 


nysql> SELECT books.title, books. bookcategory.category FROM hooks。hbookc 
ia RND 


1 title 


pmputer Ne A ens Approach ry L. Peterson 


SELECT books.tit bookcategory.category FROM books, bookc 
WHERE a id AND 
books .publi os 

! title 


alt Whitnan 


《8.98 sec) 


图 10-12 基于 一 对 一 的 表 间 关系 进行 选择 的 结果 


2. 一 对 多 的 表 间 关系 


在 讨论 一 对 多 的 表 间 关系 之 前 , 先 在 Reader 表 和 Uselog 表 中 添加 一 些 数据 , 如 图 10-13 
所 示 。 


! logid ! bookid ! reader e ! UseTs 


BOR 
BOR 
RET 


ysql> SELECT * 


aderid ! e ! 4 birthday  ! contact 


B086. 

A 

8B| 

| 
la rows in set (8-9B sec) 


Inysal> 


图 10-13 对 多 的 表 间 关系 


现在 Books 表 中 有 4 条 数据 、Reader 表 中 有 4 条 数据 ,而 Uselog 表 中 有 3 条 数据 。 在 
Uselog 表 中 ，usetype 列 的 “BOR” 代 表 的 是 “ 借 出 ”， 而 “RET” 代 表 的 是 “归还 ”。 

假如 ， 我 们 需要 查找 所 有 已 借 出 图 书 及 读者 的 相关 信息 ， 可 以 运行 如 下 的 命令 : 

SELECT reader.name, reader.contact, books.title, uselog.date 

FROM reader, books, uselog WHERE 


books.bookid = uselog.bookid AND reader.readerid = uselog.readerid AND 
uselog.usetyp = 'BOR'; 
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运行 该 命令 后 ， 系 统 会 返回 所 有 已 借 图 书 及 读者 的 相关 信息 ， 如 图 10-14 所 示 。 


图 10-14 基于 一 对 多 的 表 间 关系 进行 选择 的 结 


10.3.3 ”查询 结果 的 排序 和 组 合 


有 时 , 我 们 也 需要 对 数据 进行 排序 和 组 合 , 以 方便 数据 的 呈现 和 统计 。 现 在 就 以 Reader 

表 中 的 数据 为 例 ， 来 看 看 如 何 使 用 ORDER BY 从 名和 GROUP BY 从 句 。 
.查询 结果 的 排序 

如 果 我 们 想 要 让 Reader 表 中 的 数据 按照 读者 的 生日 从 大 到 小 进行 排序 ， 那么 , 就 可 以 
运行 如 下 的 命令 : 

SELECT * FROM reader ORDER BY birthday; 

运行 这 条 命令 后 ， 会 发 现 readerid 为 2 的 读者 Marilyn Monroe 被 放 到 了 第 一 行 ， 而 
readerid 为 1 的 读者 James Bond 被 放 到 了 第 二 行 。 

默认 情况 下 ，ORDER BY 子 句 会 按照 从 大 到 小 〈 数 量 上 ) ， 从 前 往 后 (时 间 上 ) 的 顺 
序 进行 排序 ， 使 用 ASC 关键 字 。 比 如 上 面 那 条 命令 还 可 以 写成 : 

SELECT * FROM reader ORDER BY birthday ASC; 

如 果 想 让 查询 的 结果 按照 从 小 到 大 数量 上 ) ， 从 后 往 前 (时 间 上 ) 的 顺序 进行 排序 
的 话 ， 就 得 使 用 DESC 关键 字 了 。 比 如 : 

SELECT * FROM reader ORDER BY birthday DESC; 

这 时 ， 排 在 查询 结果 最 前 面 的 一 条 记录 是 readerid 为 4 的 Juday Garland。 而 readerid 
为 2 的 Marilyn Monroe 则 排 在 了 最 后 。 


2. 查询 结果 的 分 组 


如 果 想 统计 Books 数据 表 中 类 型 为 “Literature” 和 “Computer Science” 的 图 书 各 有 多 
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少 本 ， 我 们 可 以 运行 如 下 的 命令 : 

SELECT bookcategory.category, COUNT (books .title) AS count FROM books, 

bookcategory 

WHERE books.catid = bookcategory.catid 

GROUP BY books.catid; 

在 这 条 命令 中 使 用 了 COUNTO 函 数 、AS 关键 字 和 GROUP BY 从 句 ， 其 中 COUNTO 
函数 的 功能 是 统计 符合 条 件 的 记录 一 共有 多 少 条 、AS 关键 字 可 以 将 它 前 面 的 字段 名 或 表 
达 式 改 个 名 字 ， 而 GROUP BY 从 句 则 指定 排序 的 标准 。 运 行 这 条 命令 后 ， 系 统 返回 了 
张 表 格 ， 分 别 列 出 了 类 型 为 “Literature” 和 “Computer Science” 的 图 书 各 有 多 少 本 ， 如 图 
10-15 所 示 。 


di 


图 10-15 查询 结果 的 分 组 


除了 COUNTO 函 数 之 外 ,可 以 和 GROUP BY 子 句 搭配 使 用 的 函数 还 有 :AVGO、MINO、 
MAX0O 和 SUM0O。 它 们 的 功能 分 别 是 计算 查询 结果 的 平均 值 、 最 小 值 、 最 大 值 和 总 和 。 


10.4 使 用 PHP 来 操作 数据 库 


在 上 一 节 里 通过 MySQL 的 命令 行 窗 口 学 习 了 简单 的 SQL 命令 ， 知 道 如 何 使 用 SQL 
命令 创建 数据 库 和 数据 表 、 向 数据 表 中 添加 数据 以 及 在 数据 表 中 查找 符合 指定 条 件 的 记录 。 
其 实 ， 我 们 可 以 通过 编写 PHP 脚本 来 实现 这 一 切 。 

不 知道 大 家 是 否 还 记得 ， 在 第 2 章 里 讲 了 如 何 安装 phpMyAdmin 这 款 软件 。 图 10-16 
展示 了 以 root 用 户 登 录 phpMyAdmin 的 显示 的 管理 界面 。 


图 10-16 phpMyAdmin 管理 界面 
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使 用 phpMyAdmin 可 以 很 方便 地 对 数据 库 进行 创建 数据 表 、 添 加 数据 、 修 改 数据 、 删 
除数 据 和 查找 数据 等 一 系列 操作 。 而 所 有 这 一 切 都 是 PHP 脚本 的 功劳 。 

PHP 为 我 们 提供 了 三 套 应 用 程序 接口 (API) 来 和 MySQL 数据库 进行 配合 ， 它 们 分 别 
是 MySQLi、PDO 和 MySQL。PHP 官方 推荐 我 们 使 用 前 两 种 。 按 照 官方 的 说 法 ，MySQL 
(API) 已 经 处 于 Deprecated 状态 , 也 就 是 不 再 开发 和 维护 了 。 表 10-4 所 示 比 较 了 这 三 种 应 
用 程序 接口 的 特点 。 


表 10-4 ”MySQL 数据 库 扩展 特性 比较 


开始 支持 版 本 

MySQL 5.x 是 否 提供 支持 
开发 状态 

生命 周期 

新 开发 项 目 是 否 推荐 
基于 对 象 编 程 接口 
基于 过 程 编程 接口 


是。 | 
支持 使 用 mysqlnd 执行 非 块 状 异 步 查询 | 是 | 
是 
[是 | 
[是 


长 连接 
支持 字符 集 
支持 数据 库 事务 


从 表 10-4 中 可 以 看 出 在 PHP 5 之 前 , 广泛 使 用 的 是 PHP 的 MySQL 扩展 对 MySQL 数 
据 库 进行 操作 的 ， 该 扩展 因 不 支持 基于 对 象 编程 而 被 废弃 。 在 PHP 5 发 布 时 ，PHP 为 其 增 
加 了 一 项 过 渡 性 质 的 扩展 ， 即 mysqli， 该 扩展 同时 支持 基于 对 象 编程 和 基于 过 程 编程 。 在 
发 布 PHP 5 的 第 一 个 子 版 本 时 ，PHP 又 推出 了 一 项 功能 更 加 全 面 的 数据 库 操作 接口 ， 名 为 
PHP Data Objects (PDO) ， 它 仅 支 持 基于 对 象 编程 ， 将 包括 MySQL 数据 库 在 内 的 所 有 数 
据 库 类 的 操作 都 整合 在 了 一 起 。 

在 下 面 的 例 程 中 ， 我 们 将 使 用 MySQLi 和 PDO 两 种 方式 进行 脚本 的 编写 。 在 脚本 的 
编写 中 尽量 突出 两 种 扩展 基于 对 象 的 特点 。 

好 了 ， 闲 话 少 说， 我 们 来 看 看 如 何 使 用 PHP 操作 MySQL 数据 库 吧 。 在 此 之 前 ， 先 为 
LibraryManager 数据 库 分 配 一 个 管理 员 “libarian”， 其 安全 密 钥 为 “librarian9235”。 

GRANT ALL PRIVILEGES ON librarymanager.* 


TO librarian@localhost IDENTIFIED BY 'librarian9235' 
WITH GRANT OPTION; 


在 这 条 指令 中 : 

口 GRANT 表示 进行 权限 分 配 , ALL PRIVILEGES 表示 所 有 权限 , 也 可 以 指定 具体 的 
权限 ， 如 SELECT、UPDATE 和 CREATE 等 。 具 体 可 见 表 10-5 详细 查看 MySQL 
用 户 常用 权限 。 

口 ON 后 跟随 的 是 指定 的 数据 库 以 及 数据 库 中 指定 的 表格 。 在 上 例 中 ， 我 们 指定 的 
librarymanager 数据 库 中 的 所 有 表格 。 
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口 TO 后 跟随 的 是 数据 库 用 户 名 ， 注 意 
口 IDENTIFIED BY 后 跟随 的 是 数据 库 | 
口 WITH GRANT OPTION 表示 该 用 户 可 以 分 配 权限 给 其 他 用 户 。 


E 用 户 名 后 要 加 “@” 和 主机 名 。 
户 的 密码 ， 也 就 是 之 前 提 到 的 安全 密 钥 。 


在 指定 了 这 些 参 数 之 后 ， 我 们 就 可 以 创建 一 个 MySQL 用 户 ， 并 为 其 分 配 可 管理 的 数 


据 库 和 相关 管理 权限 。 
表 10-5 _ MySQL 用 户 常用 权限 
权 限 列 Ey 4 
CREATE Create_priv 数据 库 、 表 或 索引 
DROP Drop_priv 数据 库 或 表 
GRANT OPTION Grant priv 数据 库 、 表 或 保存 的 程序 
REFERENCES References_priv 数据 库 或 表 
ALTER Alter priv 表 
DELETE Delete priv 表 
INDEX Index_priv 表 
INSERT Insert_ priv 表 
SELECT Select priv 表 
UPDATE Update priv 表 
CREATE VIEW Create_view_priv 视图 
SHOW VIEW Show view_priv 视图 
ALTER ROUTINE Alter routine priv 保存 的 程序 
CREATE ROUTINE Create routine priv 保存 的 程序 
EXECUTE Execute priv 保存 的 程序 
FILE File priv 服务 器 主机 上 的 文件 访问 
CREATE TEMPORARY TABLES | Create tmp table priv 服务 器 管理 
LOCK TABLES Lock tables_priv 服务 器 管理 
CREATE USER Create_user priv 服务 器 管理 
PROCESS Process_priv 服务 器 管理 
RELOAD Reload priv 服务 器 管理 
REPLICATION CLIENT Repl client_priv 服务 器 管理 
REPLICATION SLAVE Repl_slave_priv 服务 器 管理 
SHOW DATABASES Show_db_priv 服 
SHUTDOWN Shutdown_priv 服务 E 
SUPER Super_priv 服务 器 管理 


更 多 关于 数据 库 用 户 的 操作 指令 ， 读 者 可 以 参考 PHPKnowHow 网 站 上 的 “Managing 
MySQL Users” 一 文 。 这 篇 文章 的 URL 是 http://www.phpknowhow.com/mysql/managing- 


mysql-users/。 


10.4.1 


使 用 PHP 打开 和 关闭 数据 库 连接 


前 面 提 到 过 MySQLi 扩展 同时 支持 基于 对 象 开 发 和 基于 过 程 开 发 。 那 么 体现 在 连接 数 
据 库 上 ，MySQLi 扩展 提供 了 两 种 连接 数据 库 的 方式 。 其 中 ， 一 种 是 基于 过 程 的 数据 库 连 
接 ， 而 另 一 种 则 是 基于 对 象 的 数据 库 连 接 。 


ss 
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基于 过 程 的 数据 库 连 接 : 

$mysql = mysqli connect ($hostname, $myuser, $mypassword, $mydb); 

基于 对 象 的 数据 库 连 接 : 

Smysql = new mysql ($hostname, $myuser, $mypassword, $mydb); 

注意 如 果 在 安装 MySQL 时 修改 了 MySQL 服务 端口 。 那 么 在 这 两 条 命令 中 ， 还 得 提 
供 第 五 个 参数 ，MySQL 服务 端口 号 。 

如 果 我 们 使 用 PDO 扩展 ， 可 以 使 用 类 似 下 面 的 指令 连接 数据 库 

S$pdoMySQL = new PDO('mysql:host=hostname; dbname=mydb', $myuser, $mypass); 

创建 一 个 PDO 对 象 需要 三 个 参数 : 第 一 个 参数 指定 了 主机 名 和 数据 库 名 ， 称 为 数据 
源 字符 串 , 第 二 个 参数 指定 了 数据 库 的 管理 员 名 , 第 三 个 参数 指定 了 数据 库 管 理 员 的 密码 。 

【 例 10.14】 使 用 MySQLi 和 PDO 扩展 连接 MySQL 数据 库 。 


业 <?php 

及 // 使 用 基于 过 程 的 编程 方式 

| $conn = @mysqli connect('localhost', 'librarian', 'librarian9235', 
"librarymanager'); 

4 

5 if (mysqli connect errno($conn)) 

6 echo "Failed to connect to MySQL: '.mysqli connect error(); 

8 

9 // 使 用 基于 对 象 的 编程 方式 

10 $conn = @new mysqli('localhost', 'librarian', 'librarian9235°', 
"librarymanager'); 

并 

理 肥 if ($conn->connect errno) 

3 echo 'Failed to connect to MySQL: '. $conn->connect error; 

14 

15 

16 // 使 用 PDO_MysQL 的 连接 方式 

i try { 

18 $conn = @new pdo ('mysql:host=localhost; dbname=librarymanager', 

'librarian', 'librarian9235'); 

19 } catch (PDOException Serr) { 

20 echo $err->getMessage(); 

21 } 

2 


在 上 面 的 三 段 脚本 中 ， 我 们 发 现在 定义 数据 库 连 接 时 ， 无 论 是 使 用 MySQLi 扩展 的 基 
于 过 程 的 编程 、MySQLi 扩展 的 基于 对 象 的 编程 、 还 是 PDO 扩展 ， 都 在 定义 连接 的 语句 前 


使 用 了 “@” 


符号 。 添 加 这 个 符号 后 ， 若 该 句 存 在 错误 ， 则 可 以 避免 系统 输出 该 错误 。 这 


样 一 来 ， 我 们 就 可 以 自主 的 确定 何 时 ， 在 何 处 显示 错误 提示 了 。 

在 使 用 MySQLi 扩展 的 基于 过 程 的 编程 中 ， 使 用 了 MySQLi 扩展 为 我 们 提供 的 独立 的 
功能 函数 mysqli_connect_ermo(0)、mysqli_connect_error0 和 mysqli_connect()。 在 这 三 个 函 
数 中 ， 前 两 个 函数 各 返回 一 个 字符 串 ， 而 后 一 个 函数 返回 一 个 mysqli 对 象 。 

在 使 用 MySQLi 扩展 的 基于 对 象 的 编程 中 ， 我 们 把 与 MySQL 数据 库 服 务 器 的 连接 直 


接 定 义 成 了 一 


个 mysqli 对 象 ， 然 后 使 用 了 该 对 象 的 connect_errno 和 connect_error 属性 来 


判断 连接 是 否 成 功 。 
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而 对 于 PDO 扩展 ， 我 们 使 用 了 try...catch 结构 和 PDOException 对 象 来 判断 数据 库 连 
接 是 否 正常 。 其 中 PDOException 对 象 是 Exception 对 象 的 扩展 。 

知道 了 如 何 连接 指定 的 数据 库 , 现在 再 来 看 看 如 何 关 闭 这 个 数据 库 连 接 。 虽然 PHP 会 
在 当前 页 面 所 有 的 脚本 执行 完毕 后 ,自动 关闭 数据 库 连 接 。 但 是 PHP 的 MySQLi 扩展 基于 
过 程 的 编程 方法 还 是 为 我 们 提供 了 一 个 用 于 关闭 当前 数据 库 连 接 的 方法 
mysql_close($dbConn) 。 建 议 大 家 养 成 良好 的 习惯 ,在 不 需要 数据 库 连 接 时 使 用 
Imysql_close($dbConm) 方 法 关闭 该 连接 。 这 个 方法 只 有 一 个 参数 ， 那 就 是 已 建立 的 数据 库 连 
接 。 如 果 使 用 的 MySQLi 基于 对 象 的 编程 方法 ， 那 么 可 以 使 用 mysqli 对 象 的 close() 方 法 来 
关闭 指定 的 数据 库 连接 。 对 于 使 用 PDO 扩展 创建 的 连接 , 我 们 只 需要 将 存储 该 连接 的 变量 
设置 为 空 即 可 断 开 该 连接 。 


10.4.2 ”使 用 PHP 输出 数据 库 查询 结果 


在 这 一 步 里 ， 我 们 还 是 用 三 种 方式 分 别 来 尝试 一 下 如 何 输出 查询 结果 。 
【 例 10.1S-1】 使 用 MySQLi 扩展 基于 过 程 的 编程 输出 查询 结果 。 


Eb 
也 
咏 


on 心 


<?php 


bs 


// 和 MySQL 数据 库 服务 器 建立 连接 
$dbConn = emysqli_connect ('localhost', 'librarian', 'librarian9235', 
"librarymanager'); 
if (mysqli connect errno($dbConn)) 
die('Failed to connect to the database: '.mysqli connect_ 
error ($dbConn)); 


// 将 查询 指令 存 入 $query 变量 中 
$query = "SELECT bookid, title, author FROM books"; 


// 获 取 MysQLi 查询 结果 , 并 将 结果 存 入 $result 变量 中 ,该 变量 为 MYSQLi 结果 对 象 
$result = mysqli query ($dbConn, $query); 


// 使 用 mysqli_fetch_array() 方 法 指定 的 MysQLi 结果 对 象 转换 成 数组 ， 并 遍历 数组 
while (Sbook = mysqli fetch array ($result)){ 
echo TD “bookl0Ol." <i>.Sbookll] "</i> by ".$bookl2]: 
echo '<br>'; 


lL 


// 后 续 处 理 
mysqli free result ($result); 
mysqli close($dbConn); 


在 例 10.15-1 的 这 段 脚本 中 ， 我 们 使 用 了 基于 过 程 的 编程 。 在 编写 脚本 的 过 程 中 ， 使 
用 了 MySQLi 扩展 为 我 们 提供 的 mysqli_queryO0、mysqli_fetch_arrayO0、mysqli_free_resultO 
和 mysqli_close() 独 立 函数 。 它 们 的 作用 如 下 所 示 。 
mysqli query( 方 法 用 来 执行 SQL 查询 指令 : 该 方法 的 第 一 个 参数 为 字符 中 变量 ， 


| 


口 


返回 


的 结果 为 一 个 MySQLi 结果 对 象 。 


mysqli_fetch_array0 方 法 用 来 读 取 MySQLi 结果 对 象 中 存储 的 数据 : 该 方法 的 第 一 
个 参数 为 MySQLi 结果 对 象 ， 返 回 的 结果 为 一 个 数组 。 
mysqli free_result() 方 法 用 来 释放 MySQLi 结果 对 象 占用 的 内 存 : 该 方法 的 第 一 个 


"3s 


第 3 篇 ”PHP 编程 基础 


参数 为 MySQLi 结果 对 象 ， 返 回 的 结果 为 空 值 (NULL) 。 


口 mysqli_close0 方 法 用 来 断 开 和 MySQL 数据 库 的 连接 : 该 方法 的 第 一 个 参数 为 


MySQLi 数据 库 连 接 对 象 ， 若 连接 断 开 ， 返 回 的 结果 为 TRUE。 


我 们 也 可 以 使 用 MySQLi 基于 对 象 的 编程 来 编写 功能 相同 的 脚本 。 
【 例 10.15-2】 使 用 MySQLi 扩展 基于 对 象 的 编程 输出 查询 结果 。 


1 


WN 


<?php 


// Establish the MySQL connection 
$dbConn = @new mysqli('localhost', ‘'librarian', 'librarian9235', 
'librarymanager'); 
if ($dbConn->connect errno) 
die('Failed to connect to the database: ' .$dbConn->connect error); 


// Prepare the SQL statement 
$query = "SELECT bookid, title, author FROM books'; 


// Prepare the result for the mysql fetch array() method 
$result = $dbConn->query ($query); 


// Iterate $result 

while($book = $result->fetch array()){ 
echo “ID "Sbook[t0Ol. <i>” -Sbook[ll]e"</i>. by "Sboorl2)]s 
echo '<br>'; 

} 


// Post-process 
$result->free(); 
$dbConn->close(); 


在 这 段 脚 本 中 , 我 们 使 用 了 基于 对 象 的 编程 方法 , 把 SdbConn 定义 成 一 个 mysqli 对 象 。 
这 个 对 象 有 query0 和 close() 方 法 以 及 connect_errmo0 和 connect_error0 属 性 。 其 中 queryO 


方法 返 


回 的 值 是 个 mysqli 结果 对 象 ， 而 这 个 对 象 则 拥有 fetch_array0 和 free( 方 法 。 


现在 再 使 用 PDO 扩展 来 看 看 是 不 是 可 能 得 到 同样 的 结果 。 
【 例 10.1S-3】 使 用 PDO 扩展 输出 查询 结果 。 


1 
本 
3 


<?php 


a 


ey 
$dbConn = @new PDO('mysql:host=localhost;dbname= 
librarymanager','librarian','librarian9235°'); 


$query = "SELECT bookid, title, author FROM books'; 


foreach ($dbConn->query ($query) as Sbook) { 
echo "ID °'.$book[0]." <i>'.$book[1].'</i> by '.$book[2]; 
echo '<br>'; 
上 
} catch (PDOException Serr) { 
echo $err->getMessage(); 


运行 了 这 三 段 程序 后 ， 会 发 现 得 到 的 结果 都 是 一 样 的 : 


Tm 
D 


3 The Scarlet Letter by Nathaniel Hawthorne 
4 Moby Dick by Herman Melville 


ID 5 Leaves of Grass by Walt Whitman 
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ID 6 Computer Networks: A Systems Approach by Larry L. Peterson 


需要 注意 的 是 , 在 使 用 PDO 扩展 时 , PHP 并 没有 在 PDO 扩展 中 为 我 们 提供 与 MySQLi 
扩展 对 应 的 对 象 和 方法 ， 而 是 提供 了 一 套 自 有 的 方法 。 而 这 些 方法 通常 比 MySQLi 扩展 更 
加 高 效 。 

关于 MySQLi 扩展 提供 的 对 象 及 其 属性 和 方法 ，PHP 官方 网 站 上 提供 了 一 张 完整 的 对 
照 表格 ， 读 者 可 以 在 这 张 表 格 上 找到 MySQLi 对 象 相同 属性 和 具有 相同 功能 的 方法 在 基于 
对 象 和 基于 过 程 编程 时 的 不 同 表 现 。 具 体 的 表格 可 以 在 http://www.php.net/manual/zh/ 
mysqlisummary.php 页 面 上 找到 。 

关于 PDO 扩展 提供 的 对 象 及 其 属性 和 方法 ， 读 者 也 可 以 参考 PHP 官方 网 站 上 的 为 我 
们 提供 的 相关 说 明 。 具 体 的 网 址 是 http://www.php.net/manual/en/book.pdo.php。 


10.4.3 使 用 PHP 来 添加 、 修 改 和 删除 数据 库 数据 


在 第 三 节 里 , 我 们 学 习 了 如 何 执行 SQL 语句 来 添加 修改 和 删除 数据 库 中 的 数据 。 现在 
来 看 看 如 何 通 过 PHP 来 执行 SQL 语句 。 

PHP 的 MySQLi 和 PDO 扩展 都 支持 预 编译 语句 (Prepared Statement) 的 使 用 。 什么 是 
预 编译 语句 呢 ? 所 谓 的 预 编译 语句 指 的 就 是 一 个 执行 SQL 指令 的 流程 。 在 这 个 流程 中 , 我 
们 可 以 预先 编译 一 条 SQL 指令 ， 其 中 若干 参数 的 值 没有 指定 ,接着 绑 定 若 干 变量 到 这 些 参 
数 ， 最 后 正式 执行 这 条 语句 。 这 样 一 来 ， 语 句 的 复杂 程度 和 数量 都 有 所 提升 。 那 么 这 么 做 
有 什么 好 处 呢 ? 

按照 通常 的 认识 ， 使 用 预 编译 语句 可 以 带 来 如 下 好 处 : 

口 提高 代码 的 可 读 性 和 可 维护 性 ; 

口 提高 代码 执行 的 性 能 ; 

口 提高 了 代码 执行 的 安全 性 。 

因此 ， 大 家 在 处 理 对 数据 库 的 写 操作 时 ， 务 必 使 用 预 编译 。 这 样 即 可 以 保证 存储 在 数 
据 库 中 的 数据 安全 ， 还 可 以 实现 上 述 好 处 。 

现在 ， 我 们 一 起 来 看 看 如 何 使 用 PHP 来 添加 、 修 改 和 删除 数据 库 中 的 记录 吧 。 

首先 , 我 们 使 用 MySQLi 扩展 面向 对 象 的 编程 方法 在 数据 表 Bookcategory 中 添加 和 删 
除 一 条 记录 。 
【 例 10.16-1】 使 用 MySQLi 扩展 面向 对 象 的 编程 方法 在 指定 数据 表 中 添加 和 删除 一 条 


1 <?php 
包 $dbConn = enew mysqli('localhost','librarian','librarian9235"', 


"librarymanager'); 
if ($dbConn->connect error) { 

die('Failed to connect to the database: '.$dbConn->connect error); 
} 


// 指 定 图 书 类 目 


Scategory = "Productivity'"7 


// 准 备 SQL 查询 语句 
if($stmt = $dbConn->prepare('SELECT * FROM bookcategory WHERE 


PFAoowaewmww 


po 


ss 
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category=?'))1{ 


之 

3 // 将 指定 的 图 书 类 目 绑 定 到 准备 好 的 SQL 查询 语句 中 

14 $stmt -> bind param('s', $category); 

5 

16 // 执 行 SQL 查询 语句 

Ep $stmt -> execute(); 

18 

19 // 将 查询 结果 绑 定 到 指定 变量 中 

20 $stmt -> bind result($sCatID, $sCat); 

2 

22 // 获 取 查 询 结果 

2 $stmt -> fetch(); 

24 

25 // 关 闭 当 前 查询 

26 $stmt -> close(); 

27 

28 /* 检 查 指定 图 书 类 目 是 否 存在 。 若 存在 ， 删 除 该 类 目 。 

29 若 不 存在 ， 添 加 该 类 目 */ 

30 if (!isset($sCatID) ){ 

31 // 准 备 添加 指定 图 书 类 目 

3 有 If($stmt = $dbConn->prepare('INSERT INTO bookcategory SET 
category=2"))1{ 

33 

34 // 将 指定 图 书 类 目 绑 定 到 准备 好 的 SQL 语句 中 

四 $stmt->bind param('s', $category); 

36 

Ey // 执 行 SQL 语句 

38 $stmt->execute (); 

39 

40 // 获 取 已 插入 记录 对 应 的 catid 

41 $insertID = $stmt->insert id; 

42 

43 // 关 闭 当前 查询 

44 $stmt->close(); 

45 

46 // 输 出 语句 执行 结果 

47 Printf('"Category ss (ID: ss) has been added.', $category, 

SinsertID) ; 

48 } 

49 } else { 

50 // 准 备 删除 指定 记录 

51 if($stmt = $dbConn->prepare ('DELETE FROM bookcategory WHERE 
category=?')){ 

与 奖 $stmt->bind param('s', $category); 

53 $stmt->execute () 7 

54 printf('Category ss (ID: ss) has been deleted.', 

$category, $sCatID); 

35 } 

56 } 

Sh 此 

| 光 交 


在 上 面 这 段 脚本 中 ， 我 们 使 用 SELECT 语句 查询 指定 图 书 类 目 ， 并 根据 查询 结果 判断 
该 类 目 是 否 存在 。 有 具体 的 方法 是 : 将 查询 结果 输出 到 指定 的 变量 中 ， 然 后 使 用 issetO 函 数 
判断 该 变量 是 否 已 设置 。 如 果 已 设置 ， 则 表示 指定 图 书 类 目 己 存 在 ， 需 要 删除 。 和 否则 ， 指 
定 图 书 类 目 不 存在 ， 需 要 添加 。 
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上 面 这 段 脚 本 中 , 我 们 使 用 到 了 mysqli 对 象 和 mysqli_stmt 对 象 中 的 若干 方法 和 属性 ， 
具体 如 下 : 

(1) $dbConn->prepare($query) 

该 方法 可 以 用 来 为 SQL 查询 做 准备 工作 。 在 脚本 中 ， 我 们 使 用 prepare 方法 为 查询 、 
添加 和 删除 记录 做 好 了 准备 。 在 输入 SQL 语句 时 ， 可 以 使 用 “?” 占 位 符 埋 入 若干 未 指定 
的 参数 ， 然 后 使 用 mysqli_stmt 对 象 的 bind_param 方法 指定 这 些 参数 。 

(2) $stmt->bind param($types, $param!l,[$param?,...]) 

该 方法 可 以 指定 prepare 语句 中 使 用 “?” 占 位 符 进入 的 未 指定 的 参数 。 

(3) $stmt->execute() 

该 方法 可 以 执行 准备 好 的 SQL 语句 。 需 要 注意 的 是 ，execute( 方 法 的 使 用 对 象 只 能 是 
mysqli_stmt 对 象 。 

(4) $stmt->bind result($paraml,[ $param?2,...]) 

该 方法 可 以 将 查询 结果 中 各 字段 的 内 容 绑 定 到 指定 的 变量 或 数组 中 。 

(5) $stmt->fetch() 

该 方法 用 来 获取 已 经 绑 定 到 指定 的 变量 或 数组 中 的 查询 结果 。 

(6) $stmt->insert id 

该 属性 可 以 用 来 获取 上 一 条 插入 记录 的 ID。 这 一 属性 对 于 有 着 自动 增加 ID 字段 的 数 
据 表 来 说 ， 是 十 分 有 用 的 。 

现在 ， 我 们 再 使 用 MySQLi 扩展 的 面向 过 程 的 编程 方法 重新 编写 例 10.16-1 中 的 这 段 
脚本 。 

【 例 10.16-2】 使 用 MySQLi 扩展 面向 过 程 的 编程 方法 在 指定 数据 表 中 添加 和 删除 一 条 


是 <?php 
2 $dbConn = @mysqli connect('localhost', 'librarian', 'librarian9235', 
"librarymanager'); 
EE: if (mysqli connect error($dbConn)){ 
4 die('"Failed to connect to the database: '.mysqli connect 
error ($dbConn)); 


5 } 

6 

$category = "Biography"; 
8 

9 


If($stmt =mysqli prepare ($dbConn, 'SELECT * FROM bookcategory WHERE 
category=?')){ 


10 mysqli stmt bind param($stmt, 's', $category); 

El mysqli stmt execute ($stmt); 

2 mysqli stmt bind result($stmt, $sCatID, $sCat); 

3 mysqli stmt fetch($stmt); 

14 mysqli stmt close($stmt); 

Ls if(!isset ($sCatID)){ 

16 if($stmt = mysqli prepare ($dbConn, 'INSERT INTO bookcategory 
SET category=?"')){ 

Wy mysqli stmt bind param($stmt, 's', $category); 

18 mysqli stmt execute ($stmt); 

19 $insertID = mysqli stmt insert id($stmt); 

20 mysqli stmt close($stmt); 

21 printf('Category $s (ID: ss) has been added.', $category, 

$insertID); 
Pd » 
3 } else { 


a 
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24 


25 
26 
金池 


28 
29 
30 
> 


if($stmt = mysqli prepare ($dbConn, 'DELETE FROM bookcategory 
WHERE category=2") )1{ 
mysqli stmt bind param($stmt, 's', $category); 
mysqli stmt execute($stmt); 
printf('Category ss (ID: $s) has been deleted.', 
$category, $sCatID); 


在 上 面 这 段 脚本 里 ， 我 们 使 用 了 MySQLi 扩展 的 面向 过 程 的 编程 方法 ， 实 现 了 与 例 
10-16-1 中 的 脚本 同样 的 功能 。 这 段 脚本 中 使 用 到 了 MySQLi 扩展 对 象 的 如 下 几 个 方法 和 


属性 。 


(1) mysqli prepare($dbConn, $query) 方 法 : 相当 于 $dbConn->prepare($query)。 

(2) mysqli_stmt_bind param($stmt, $type, Sparaml[, Sparam2, ...]) 方 法 : 相当 于 $stmt-> 
bind_param($type, Sparam1[, Sparam2, ...])。 

(3) mysqli_stmt_ bind result($stmt，$paraml[，$param2，...]) 方 法 : 相当 于 $stmt->bind 
Iesult($paraml[, $param2, ...])。 

(4) mysqli_stmt_execute($stmt) 方 法 : 相当 于 $stmt->execute()。 

(5) mysqli_stmt_fetch($stmt) 方 法 : 相当 于 $stmt->fetch()。 

(6) mysqli_ stmt_insert id($stmb 属 性 : 相当 于 $stmt->insert id;。 

下 面 我 们 来 使 用 PDO 扩展 修改 指定 数据 表 中 的 指定 记录 。 
【 例 10.16-3】 使 用 PDO 扩展 修改 指定 数据 表 中 的 指定 记录 。 


<?php 


try { 


$dbConn = new PDO('mysql:host=localhost;dbname=librarymanager', 
"librarian', librarian9235')s 


"Linguistics'7 
"Language Studies'; 


$category 
$changeTo 


if($stmt = $dbConn->prepare('SELECT * FROM bookcategory WHERE 
category=:category')){ 
$stmt->bindParam(':category', $category, PDO::PARAM STR); 
$stmt->execute(); 
$stmt->bindColumn (1, $sCatID); 
$stmt->fetch(); 
if(!isset ($sCatID)){ 
printf('Category $s does not exist.', $category); 
} else { 
if($stmt = $dbConn->prepare('UPDATE bookcategory 
SET category=:changeto WHERE catid=:catid')){ 
$stmt->bindParam(':changeto', $changeTo, PDO::PARAM STR); 
$stmt->bindParam(':catid', $sCatID, PDO::PARAM INT); 
$stmt->execute (); 
printf('Category ss has been changed to %s', $category, 
$changeTo); 


} 
} 


} catch (PDOException S$err) { 


有 


公 六 


= 


echo S$err -> getMessage(); 
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执行 这 段 脚 本 运行 后 ,原来 名 为 “Linguistics” 的 图 书 类 目 变 成 了 “Language Studies”。 
在 这 段 脚本 中 ， 我 们 首先 查找 指定 的 图 书 类 目 。 通 过 判断 该 类 目 是 否 存在 来 决定 下 一 步 执 
行 的 脚本 : 若 存在 ， 则 执行 修改 ; 若 不 存在 ， 则 输出 通知 。 

编程 中 ， 我 们 使 用 了 PDO 扩展 的 PDO 对 象 和 PDOStatement 对 象 中 的 若干 方法 如 下 
所 示 。 

(1) $stmt->bindParam($param, $variable, [$datatype]) 方 法 

该 方法 一 次 只 能 绑 定 一 对 参数 与 变量 。 除 了 使 用 之 前 提 到 的 “?” 占 位 符 之 外 ， 我 们 还 
可 以 使 用 名 称 占 位 符 ， 名 称 占 位 符 的 格式 为 “:name”。 本 例 使 用 了 后 者 。 若 需要 使 用 “?” 
占 位 符 ， 在 指定 Sparam 参数 时 ， 可 输入 代表 查询 语句 中 该 参数 所 处 位 置 的 正 整 数 。 该 方法 
相当 于 MySQLi 扩展 中 的 mysqli_stmt bind param() 方 法 。 

(2) $stmt->bindColumn($column, $variable) 方 法 

该 方法 一 次 只 能 绑 定 一 对 字段 名 与 变量 。 在 执行 了 准备 好 的 SQL 语句 之 后 , 我 们 可 以 
将 查询 结果 绑 定 到 若干 变量 中 。 其 中 ，S$column 可 以 是 字段 名 ， 也 可 以 是 该 字段 在 数据 表 
中 所 处 位 置 的 正 整 数 。 本 例 使 用 了 后 者 。 该 方法 相当 于 MySQLi 扩展 中 的 
mysqli _stmt bind result( 方 法 。 

(3) $stmt->execute() 方 法 

该 方法 执行 准备 好 的 SQL 语句 。 相 当 于 MySQLi 扩展 中 的 mysqli_stmt_execute() 方 法 。 

(4) $stmt->fetch0 方 法 

该 方法 获取 SQL 语句 执行 后 的 结果 集 。 相 当 于 MySQLi 扩展 中 的 mysqli_stmt fetch() 
为 法 5 


10.5 ”实战 练习 : 记 账 工具 (下 ) 


在 前 两 章 中 , 我 们 为 记 账 工具 编写 了 一 个 名 为 BookKeeping 的 PHP 类 , 并 且 搭 建 好 了 
工具 的 用 户 界面 。 在 本 节 里 ， 将 为 该 记 账 工具 规划 数据 库 、 制 定 批量 导入 功能 中 使 用 的 模 
板 文件 、 并 为 这 些 页 面 添加 相应 的 功能 。 现 在 就 让 我 们 一 步 一 步 来 实现 奇迹 吧 。 


10.5.1 规划 数据 库 


通过 之 前 规划 的 用 户 界 面 ， 我 们 的 数据 库 主 要 包括 以 下 几 张 表格 。 图 10-17 展示 了 数 
据 库 中 包含 的 数据 表 以 及 数据 表 之 间 的 关系 。 


gender 
password 


userld 


图 10-17 记 账 工具 数据 库 表 及 表 间 关系 


Ns 
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由 于 规划 的 功能 都 比较 基础 ,所 以 数据 表 不 多 , 只 有 三 张 , 它们 分 别 是 User 表 、Incomes 
表 和 Expenses 表 。 其 中 ，Incomes 和 Expenses 两 张 表 中 都 有 一 个 名 为 UserId 的 外 键 ， 表 明 
了 这 三 张 表 之 间 的 关系 。 

读者 可 以 根据 10.3.1 小 节 学 习 到 的 知识 在 MySQL 服务 器 上 创建 一 个 数据 库 
bookkeeping， 然 后 在 数据 库 中 创建 这 三 张 表 格 。 


10.5.2 ”批量 导入 模板 


由 于 我 们 批量 导入 的 时 候 需 要 让 用 户 在 模板 中 填写 若干 条 记录 。 然 后 ， 通 过 上 传 该 模 
板 文件 ， 批 量 导入 若干 条 记录 。 在 本 章 学 习 了 如 何 使 用 文本 文件 和 XML 文件 存 取 数 据 。 
考虑 到 我 们 规划 的 功能 比较 简单 ， 所 以 在 这 里 使 用 文本 文件 来 规划 这 个 模板 。 

在 该 文本 文件 中 ， 用 户 只 需要 输入 类 似 下 面 的 内 容 就 可 以 完成 一 条 记录 : 


{EIDE} 手 机 1 部 
{EIAM}2588.63 
{EICA} 用 


{IIDE}7 月 工资 
{IIAM}8566.63 
{IICA} 工 资 


{EIDE} 午 饭 

{EIAM}16.50 

{EICA} 食 

上 面 这 个 模板 文件 定义 了 三 条 账 务 记录 。 其 中 ， 第 一 条 和 第 三 条 为 支出 记录 ， 第 二 条 
为 收入 记录 ， 两 条 记录 之 间 用 “.…” 分 隔 。 我 们 将 每 一 条 记录 的 三 个 字段 分 别 写 入 三 行 中 ， 
并 以 “{*|**}” 来 标识 每 一 行 数据 代表 的 意义 。 比 如 ，“ {EIDE}” 标 识 了 一 条 支出 记录 的 
说 明 ， 而 “人 AM} ”标识 了 一 条 收入 记录 涉及 的 金额 。 

大 家 可 以 将 上 面 的 账 务 记录 复制 至 一 个 文本 文件 中 。 需 要 注意 的 是 ， 如 果 读 者 使 用 的 
是 Windows 操作 系统 ， 在 保存 时 ， 务 必用 UTF-8 的 编码 格式 。 操 作 方 法 如 图 10-18 所 示 。 


上 
站 Er 


| Sn - 国 er 
Flename 5 
Sope [Tet Docmerts C00) 


-| 
Sr [ee ad)| Es Bees 


图 10-18 在 Windows 操作 系统 中 使 用 UTF-8 编码 保存 文本 文件 
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10.5.3 为 页 面 添加 功能 前 的 准备 工作 


在 为 各 页 面 添加 功能 之 前 ， 再 来 考量 一 下 第 8 章 编 写 的 那个 bookKeeping 类 。 在 那个 
类 中 ， 定 义 了 两 个 私有 方法 ， 分 别 为 AddIncomeToDatabase0 和 AddExpenseToDatabase()。 
现在 我 们 就 来 编写 这 两 个 私有 方法 。 


private function addExpenseToDatabase ($desc, $amount, $category, $userId) { 


} 


// 获 取 数 据 库 连 接 
$dbconn = $this -> getDBconn(); 
// 设 置 数据 库 字符 集 ， 用 于 处 理 中 文字 符 
$dbconn->set charset ('utf8°'); 
// 使 用 sprintf () 函数 拼接 SQL 查询 语句 
$q = sprintf('SELECT expenseId FROM expenses 
WHERE eDesc="%s" AND eAmount =%s AND eCategory = "%s"', 
$desc, $amount, $category); 
// 若 上 述 拼接 的 SQL 查询 语句 有 结果 ， 则 取出 结果 ; 否则， 中 止 脚本 
if($stmt = $dbconn->prepare ($q) ) { 
$stmt->execute(); 
$stmt->bind result ($expenseId); 
$stmt->fetch(); 
$stmt->close(); 
} else { 
die('NOK'); 
} 
// 若 未 查 到 满足 条 件 的 expenseId， 开 始 向 数据 库 中 插入 记录 
if(!isset ($expenseId)){ 
S$q = sprintf('INSERT INTO expenses (eDesc,eDate,eAmount, 
eCategory, userId) 
VALUES ("%s",CURRENT DATE, $f,"%s",%d)',$desc,$amount, 
$category, $userId); 


if($stmt = $dbconn->prepare ($9q)){ 
$stmt->execute (); 
$stmt->close(); 
} else { 
die('NOK'); 
} 
// 插 入 记录 完成 后 ， 返 回 'Added" 
return 'Added'; 
} else { 
// 若 查找 到 满足 条 件 的 expenseId， 则 跳 过 插入 记录 ， 直 接 返 回 'Denied' 
return "Denied' 7 
} 
// 关 闭 数据 库 连接 
$dbconn—>kill (); 
$dbconn->close(); 


Private function addIncomeToDatabase ($desc, $amount, $category, 
SuserId) { 


// 获 取 数据 库 连接 


$dbconn = $this -> getDBconn(); 


// 设 置 数据 库 字符 集 


Sdbconn->set charset ('utf8°'); 
// 使 用 sprintf() 函数 拼接 SQL 查询 语句 
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$q = sprintf('SELECT incomeId FROM incomes 
WHERE iDesc="%s" AND iAmount =%s AND iCategory = "%s"', $desc, 
Samount, $category); 


// 若 上 述 拼 接 的 SQL 查询 语句 有 结果 ， 则 取出 结果 ; 否则 ， 中 止 脚本 
if($stmt = $dbconn->prepare ($q)){ 
$stmt->execute () 7 
$stmt->bind result ($incomeId); 
$stmt->fetch(); 
$stmt->close(); 
} else { 
die('NOK'); 
. 


// 若 未 查 到 满足 条 件 的 incomeId， 开 始 向 数据 库 中 插入 记录 

if(!isset ($incomeId)){ 
$q = sprintf('INSERT INTO incomes (iDesc,iDate,iAmount, 
iCategory, userId) 
VALUES ("%s",CURRENT DATE, $f,"%s",%d)',$desc,$amount, 
$category, $userId); 


if($stmt = $dbconn->prepare($q)){ 
$stmt->execute (); 
$stmt->close(); 

} else { 
die('NOK'); 

} 


// 插入 记录 完成 后 ， 返 回 'Added'。 
return "Rdded' 7 
} else { 
// 车 查找 到 满足 条 件 的 ijncomeId， 则 跳 过 插入 记录 ， 直 接 返 回 ' Denied' 


return 'Denied'; 


} 


$dbconn->kill (); 
$dbconn->close(); 


} 


紧 接着 ， 我 们 还 需要 为 BookKeeping 类 定义 两 个 新 的 私有 方法 ， 分 别 用 来 删除 指定 的 
消费 和 收入 记录 。 这 两 个 私有 方法 定 为 removeExpenseFromDatabase() 和 removeIncomeFrom- 
Database0， 代 码 如 下 : 


private function removeExpenseFromDatabase ($expenseID) { 
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$dbconn = $this -> getDBconn () 7 
$dbconn -> set charset ('utf8°'); 


$q = sprintf('DELETE FROM expenses WHERE expenseID = $s', $expenseID); 
try { 
$stmt = $dbconn->prepare($q); 
$stmt->execute(); 
} catch (Exception $e) { 
Smsg = "Failed"; 
Sel 
} 


$dbconn—>kill(); 
$dbconn->close(); 


Smsg = "Removed"; 


第 10 章 数据 的 存储 


private function removeIncomeFromDatabase ($incomeID) { 
$dbconn = $this -> getDBconn(); 
$dbconn -> set charset ('utf8') > 


$q = sprintf('DELETE FROM incomes WHERE incomeID = $s', $incomeID); 
Ery 
$stmt = $dbconn->prepare($q); 
$stmt->execute (); 
} catch (Exception $e) { 
Smsg = "Failed"; 
die(); 
} 


$dbconn-—>kill (); 
$dbconn->close (); 


$msg = "Removed"; 


} 

除了 上 述 四 个 私有 方法 之 外 ， 还 需要 定义 两 个 用 于 修改 现 有 消费 和 收入 记录 的 私有 方 
法 。 这 两 个 私有 方法 定名 为 modifyExpenseInDatabase() 和 modifyIncomeInDatabase()。 它 们 
的 代码 如 下 : 


private function modifyExpenseInDatabase ($expenseID, $desc, $amount, 
$category) { 

$dbconn = Sthis -> getDBconn(); 

$dbconn -> set charset ('utf8°'); 


$q = sprintf('UPDATE expenses SET eDesc = "%s", 
eAmount = %f, 
SeCategory = "%s" WHERE expenseID = %s', $desc, $amount, 
$category, $expenseID); 


try 
$stmt = $dbconn->prepare($q); 
$stmt->execute(); 
} catch (Exception $e) { 
Smsg = "Failed"; 
die(); 
} 


$dbconn -> kill(); 
$dbconn -> close(); 


$msg = "Modified"; 


return $msg; 


} 


private function modifyIncomeInDatabase ($incomelID, $desc, $amount, $category){ 
$dbconn = $this -> getDBconn(); 
$dbconn -> set charset ("utf8"); 


$q = sprintf('UPDATE incomes SET iDesc = "%s", 
iAmount = %f, 
S$iCategory = "%s" WHERE incomeID = $s', $desc, $amount, 
Scategory, $incomeID); 


try { 
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$stmt = $dbconn->prepare($q); 
$stmt->execute(); 
} catch (Exception Se) { 
Smsg = "Failed"; 
die(); 
} 


$dbconn -> kill(); 
$dbconn -> close(); 


$msg = "Modified"; 


return $msg; 


| 


为 了 方便 在 页 面 上 引用 上 述 四 个 私有 方法 ， 我 们 创建 四 个 新 方法 ， 用 来 修改 和 删除 用 
户 指 定 的 数据 ， 并 在 这 四 个 新 方法 中 引用 上 述 四 个 私有 方法 。 这 四 个 新 方法 如 下 所 示 。 

(1) ModifyExpenseEntry/ModifyIncomeEntry 

在 这 两 个 新 方法 中 ， 我 们 对 用 户 提交 的 某 记 录 修 改 后 的 数据 进行 检查 ， 并 将 符合 要 求 
的 数据 存 入 数据 库 的 指定 记录 。 脚 本 如 下 : 


function ModifyExpenseEntry(S$expenseId,S$desc,S$amount,Scategory) { 
if(mb strlen($desc, 'UTF-8') <= 70 && 
preg match('/^\d+(\.\d+)?$/',$amount) && 
strlen(S$userId) > 0) { 
Snote = $this -> modifyExpenseInDatabase ($expenselId, S$desc, 
$amount, $category); 
if($note == "modified"){ 
return ' 修 改 消费 记录 【 消 "' .sprintf ('%05s', $expenseId) .'")】 成 功 。'; 
} else { 
return' 修 改 消费 记录 【 消 "' .sprintf('%05s', $expenseId) .'"】 失 败 。'; 
} 
} 
| 


function ModifyIncomeEntry ($incomeld, $desc, $amount, $category){ 
if(mb strlen($desc, 'UTF-8') <= 70 && 
preg match('/^\d+(\.\d+)?$/',$amount) &E& 
strlen($userId) > 0) { 
Snote = Sthis -> modifyIncomeInDatabase ($incomeId, S$desc, 
$amount, $category); 


if(Snote == "modified"){ 
return ' 修 改 收入 记录 【EXP"' .sprintf('%$05s', $expenseId) .'")】 成 功 。'; 
} else { 


return ' 修 改 收入 记录 【EXP"' .sprintf('%$05s',$expenseId) .'"】 失 败 。'; 
} 
3 

} 

(2) DeleteExpenseEntry/DeleteIncomeEntry 

在 这 两 个 新 方法 中 ， 我 们 将 查询 用 户 欲 删除 的 记录 是 否 存在 。 只 有 当 用 户 指定 的 记录 
存在 时 ， 才 会 引用 removeExpenseFromDatabase 和 removeIncomeFromDatabase 两 个 私有 方 
法 删除 指定 的 记录 。 脚 本 如 下 : 


function DeleteExpenseEntry(SexpenseId) { 
$dbconn = $this -> getDBconn () 7 
$dbconn->set charset ('utf8°'); 


。244 。 


第 10 章 数据 的 存储 


$q = sprinf ("SELECT * FROM expenses WHERE expenseId = %s", $expenseId); 


Smsg = Vrs 


tryt{ 
$stmt = $dbconn->prepare($q); 
$stmt->execute(); 
$stmt->store result(); 


if($stmt->num rows() == 1) { 

$note = $this->removeExpenseFromDatabase ($expenseId); 

if($note == 'removed'){ 
$msg .= ' 删 除 消费 记录 【EXP"' .sprintf('%05s', $expenseId).'"】 
成 功 。"'; 

} else { 
Smsg .= ' 删 除 消费 记录 【EXP"' .sprintf('%05s', $expenseId).'")】 
失败 。"; 


} 


} 
} catch (Exception $e)f{ 


Smsg .= ' 系 统 内 部 错误 ， 请 尽快 报告 管理 员 。' ; 


$dbconn -> kill(); 
$dbconn -> close(); 


return $msg; 


: 


function DeleteIncomeEntry ($incomeId){ 
$dbconn = $this -> getDBconn(); 
$dbconn->set charset ('utf8°'); 


$q = sprinf ("SELECT * FROM incomes WHERE expenseId = %s", $expenselId); 
Smsg = 7 


try{ 
$stmt = $dbconn->prepare ($q); 
$stmt->execute(); 
$stmt->store result(); 


if($stmt->num rows() == 1) { 
Snote = $this->removeIncomeFromDatabase (SincomeId) ; 
if(Snote == 'removed'){ 
$msg -= "删除 收入 记录 【INC"' .sprintf('%05s',$incomeId).'"】 
成 功 。'; 
} else { 
$msg -= ' 删 除 收 入 记录 【INC"' .sprintf('%05s',$incomeId).'"】 
Sr 
1 
} 


} catch (Exception Se) { 
$msg .= "系统 内 部 错误 ， 请 尽快 报告 管理 员 。 " ; 
} 


$dbconn -> kill(); 
$dbconn -> close(); 


return S$msg; 
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另外 ， 为 了 获取 用 户 指定 记录 的 详细 信息 ， 还 定义 了 一 个 SearchForRecords 方法 和 
GetDetails 方法 ， 前 者 需要 用 户 指定 日 期 范围 和 记录 类 型 ， 后 者 则 需要 用 户 指 定 记录 类 型 
和 记录 编号 。 这 两 个 方法 的 脚本 如 下 所 示 。 

(3) SearchForRecords 方法 


function SearchForRecords ($setYear, $setMonth, $recordType, $userId){ 
// 连 接 数据 库 
$dbconn = $this -> getDBconn () 7 
$dbconn->set charset ('utf8"'); 


// 计 算 时 间 点 ， 基 于 此 查询 数据 库 中 的 记录 
$setDate = date('Y-m-d',mktime (0,0,0,intval ($setMonth),1,intval 
($setYear))); 


// 定义 一 个 空 数组 


$records = array(); 


// 根据 用 户 需 要 查找 的 记录 类 型 在 数据 库 中 查找 并 输出 记录 

if($recordType == 'EXP'){ 
$q = sprintf('SELECT * FROM expenses WHERE eDate < "%s" AND userId 
= %s', $setDate, $userId); 


if($stmt = $dbconn -> prepare($q)){ 
$stmt -> execute(); 
$stmt->bind result ($expenselId, $eDesc, S$eDate, $eAmount, 
$eCategory, $userId); 
$stmt->store result(); 


if ($stmt->num rows > 0) { 


$i=0; 
while ($stmt->fetch()) { 
Srecords [$i] [0] = $expenselId; 
Srecords [Si] [1] = $eDesc; 
$records[$i] [2] = number format ($eAmount, 2); 
Srecords [$i] [3] = $eCategory; 
Srecords [Si] [4] = $eDate; 
训 主 二 汪 坟 
} 
} 
} else { 


$q = sprintf('SELECT * FROM incomes WHERE iDate < "%s" AND userId 
= %s', $setDate, $userId); 


if($stmt = $dbconn -> prepare($q)){ 
$stmt -> execute(); 
$stmt->bind result ($incomeId, $iDesc, $iDate, $iAmount, 
$iCategory, $userId); 
$stmt->store result(); 


if ($stmt->num rows > 0) { 


$i=0; 
while ($stmt->fetch()) { 
Srecords [Si] [0] = $incomelId; 
Srecords [Si] [1] = $iDesc; 
Srecords [Si] [2] = number format ($iAmount, 2); 
Srecords [Si] [3] = $iCategory; 
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Srecords [$i][4] = $iDate; 
六 下 


; 


return S$records; 


1 


(4) GetDetails 方法 
function GetDetails ($recordType, $recordId){ 


$dbconn = $this -> getDBconn(); 
$dbconn -> set charset ('utf8°'); 


$result = array(); 


if($recordType == 'expense'){ 
$q = sprintf('SELECT * FROM expenses WHERE expenseId = %s', 
SrecordId) 


if($stmt = $dbconn->prepare($q)){ 
$stmt->execute(); 
$stmt->bind result ($expenselId, $eDesc, $eDate, $eAmount, 
$eCategory, $userId); 


while ($stmt -> fetch()) { 


$result[0] = $eDesc; 
$result[1] = number format ($eAmount, 2); 
$result[2] = $eCategory; 
$result[3] = $eDate; 
} 
} else { 


$q= sprintf('SELECT * FROM incomes WHERE incomeId = %s', $recordId); 
if($stmt = $dbconn->prepare($q)){ 
$stmt->execute(); 
$stmt->bind result ($incomelId, $iDesc, $iDate, $iAmount, 
$iCategory, $userId); 


while($stmt->fetch()){ 


$result[0] = $iDesc; 

$result[1] = number format ($iAmount, 2); 
$result[2] = $iCategory; 

$result[3] = $iDate; 


上 


return $result; 


} 

在 上 述 与 数据 库 操作 相关 的 方法 中 ， 我 们 都 使 用 了 一 个 名 为 getDBconn 的 私有 方法 。 
接 下 来 , 定义 这 个 名 为 getDBconn0 的 私有 方法 用 于 获取 与 数据 库 之 间 的 连接 。 另 外 ,在 批 
量 上 传 的 方法 中 ,我 们 还 使 用 了 一 个 名 为 readTempO 的 方法 用 于 处 理 用 户 上 传 的 模板 文件 。 
这 两 个 方法 的 脚本 如 下 : 


private function getDBconn(){ 
$dbconn = @new mysqli ("localhost', 'root', 'penpaper220', 'bookkeeping'); 
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if(S$dqbconn->connect errno) 
die('" 连 接 数据 库 失 败 : ' .$dbconn->connect error) ; 
} 


return $dbconn; 


j: 


// 读 取 已 上 传 文件 的 内 容 ， 并 导入 文件 中 的 帐 务 记录 
function readTemp ($file, $userId){ 
// 判 断 用 户 是 否 已 经 成 功 上 传 了 模板 文件 。 若 是 ， 则 将 文件 中 的 内 容 按 行 读 取 到 $1ines 数组 中 
if(file exists("upload\\".S$file)){ 
$fh = fopen('upload\\' .$file, 'r'); 
$lineCount = 0; 
while (!feof ($fh)){ 
if($lineCount == 0){ 
$lines[$lineCount] = trim(substr (fgets ($fh), 3)); 
} 
else { 
$lines[$lineCount] = trim(fgets ($fh)); 
} 


$lineCount++; 
} 
// 设 置 一 些 参数 
Sorce 0 // 记 录 模 板 文件 中 的 错误 数 
$eCount = 0; // 记 录 模板 文件 中 可 导入 的 消费 记录 数 
Seonnee 0 // 记 录 模 板 文件 中 可 导入 的 收入 记录 数 
$msg = 7; //readTemp 方法 返回 消息 


// 开 始 检查 模板 文件 ， 记 录 文 件 中 的 错误 数 
for($i=0;S$i<count ($lines) ; $i=$i+4) { 
LE(SLLnes lS <> oe ou)t 
Serrtt+; 
| 


} 
// 若 错误 数 为 0， 则 开始 截取 需要 的 数据 ， 并 引用 之 前 定义 的 两 个 私有 方法 将 数据 添加 到 库 中 
if($err == 0){ 


/* 模 板 文件 中 的 各 条 记录 间 用 “.….” 隔 开 ， 每 个 分 隔行 下 三 行为 一 条 账 务 记录 。 所 
以 下 面 循 环 的 步 长 为 $i = Si + 4 */ 


for ($i=1; $i<sizeof ($lines); $i=$i+4){ 


// 检 查 每 个 分 隔行 下 三 行 的 起 始 六 个 字符 是 否 分别 为 {E1DE}、{EIAM} 和 


{EICA} 

if(substr ($lines[$i],0,6) == '{EIDE}' &é& 
substr ($lines[$i+1],0,6) "{EIAM}' && 
substr ($lines[$i+2],0,6) == '{EICA}'){ 


// 若 是 ， 则 获取 各 行 第 6 个 字符 之 后 的 内 容 存 入 对 应 的 变量 中 
$desc = mb substr ($lines[$i],6); 

Samount = mb substr($lines[$i+1],6); 

$category = mb substr($lines[$i+2],6); 


// 然 后 使 用 私有 方法 addExpenseToDatabase 将 记录 添加 到 库 中 
Sm = $this->addExpenseToDatabase ($desc, $amount, 
$category, $userId); 
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// 若 addExpenseToDatabase 返回 值 为 "Added"，$eCount 值 加 1 


if($m == "Added'){ 
SeCount++7 
} 
}else if(substr($lines[$i],0,6) == '{IIDE}' && 
substr ($lines[$i+1],0, 6) '{IIAM}' && 
substr ($lines[$i+2],0,6) == "{IICA}') { 


$desc = mb substr ($lines[$i],6); 
$amount = mb substr($lines[$i+1],6); 
$category = mb_substr ($lines[$i+2], 6); 


Sm = $this -> addIncomeToDatabase ($desc, $amount, 
$category, $userId); 


if($m == 'Added'){ 
$iCount++; 
} 


} 
$msg .= ' 总 共 添 加 了 ' .$eCount.' 条 消费 记录 和 ' .$iCount .' 条 收入 记录 。'; 


}elsef{ 
$msg .= ' 模 板 中 存在 ' .$err.' 处 错误 。 导 入 失败 。"'; 
} 
return $msg; 
} else { 
return 'NOK'; 
} 


最 后 ， 我 们 向 bookKeeping 类 中 再 添加 一 个 新 的 方法 ， 用 来 验证 用 户 的 身份 ， 名 字 为 
authenticate()。 脚 本 如 下 : 


function authenticate ($username, $password) { 
// 获 取 数 据 库 连 接 
$dbconn = Sthis -> getDBconn(); 
// 查 找 数据 库 中 是 否 存在 用 户 名 和 密码 匹配 的 记录 
$q = sprintf('SELECT userId FROM users where 
name = "ss" and password = "%s";',$username, $password); 
// 若 上 述 拼接 的 SQL 语句 可 以 运行 ， 则 运行 该 语句 获取 指定 用 户 的 用 户 ID 
if($stmt = $dbconn->prepare($q)){ 
$stmt->execute(); 
$stmt->bind result ($id); 
$stmt->fetch(); 
$stmt->close(); 


上 
// 若 查找 到 指定 用 户 的 ID， 返 回 该 ID 
if(isset($id)){ 
return $id; 
} else { 
// 和 否则 返回 “NOK”。 
return "NOK "7 
} 


// 关 闭 数据 库 连接 


$dbconn—>kill(); 
$dbconn->close(); 
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现在 ,我 们 就 可 以 使 用 bookKeeping 类 的 authenticate 方法 来 验证 用 户 身份 了 。 还 记得 
在 9.5.2 小 节 中 在 登录 页 面 上 添加 了 一 段 PHP 脚本 吗 ? 使 用 这 段 脚本 用 来 判断 用 户 提交 的 
信息 ， 并 据 此 将 用 户 转向 到 指定 的 页 面 。 这 段 脚本 如 下 : 
<?php 
session start(); 
if(isset($ REQUEST['login'])){ 


$user = $ REQUEST['usr']; 
Spws = $ REQUEST['psw']; 


ifl(strlen($user) > 0) { 
$ SESSION['user'] = $user; 
header ('location:./08.2 Book Keeping AddExpense.php'); 


: 
2> 


在 这 段 脚本 中 ， 我 们 只 是 判断 了 用 户 是 否 输入 了 用 户 名 ， 并 没有 通过 查询 数据 库 来 判 
断 数据 库 中 是 否 存在 该 用 户 。 现 在 来 修改 一 下 脚本 : 


if(isset($ REQUEST['1login"])){ 
$user = $ REQUEST['usr']; 
$pws = $ REQUEST['psw']; 
// 新 建 bookKeeping 类 对 象 
S$bk = new bookKeeping(); 
// 使 用 $bk 对 象 的 authenticate () 方法 来 获取 指定 用 户 的 用 户 ID 
$userId = $bk->authenticate ($user, $pws); 
// 根据 authenticate () 方法 返回 的 内 容 决定 用 户 的 去 向 


if($userId == 'NOK')1{ 
header ('1location:./08.1 Book Keeping Login.php'); 
} else { 


$ SESSION['user']=$user; 
$ SESSION['userId']=$userId; 
header('location:./08.2 Book Keeping AddExpense.php'); 


| 
在 上 面 这 段 脚本 中 ， 我 们 先 定 义 了 一 个 名 为 $bk 的 bookKeeping 类 的 对 象 ， 然 后 使 用 
$bk 对 象 的 es 了 用 户 是 否 存 在 ， 并 根据 该 方法 的 返回 值 判断 用 户 是 否 


可 以 进入 其 他 页 面 
现在 , 我 们 使 用 phpMyAdmin 先 在 数据 库 的 user 表 中 添加 一 条 记录 , 如 图 10-19 所 示 。 


TT | 
9 mr OO 


从 Fos D2001 /71001 / bookseepeg /oer | pte 


图 10-19 向 数据 库 的 user 表 中 添加 一 条 用 户 记录 
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接着 , 在 浏览 器 中 打开 制作 的 记 账 工具 的 登录 页 面 , 在 用 户 名 和 密码 处 分 别 输入 “user” 
和 “Passw0rd”。 然 后 单 击 “立即 登录 ”按钮 ， 如 图 10-20 所 示 。 


pe 


9 - TT 


图 10-20 登录 记 账 工具 
到 此 ， 我 们 就 完成 了 为 页 面 添加 功能 前 的 准备 工作 。 


10.5.4 为 页 面 添加 功能 


1. 为 “添加 消费 记录 ”页 面 添加 功能 代码 

在 第 9 章 里 ， 在 添加 消费 记录 页 面 上 添加 了 一 张 表单 。 表 单 中 有 一 个 文本 域 、 一 个 下 
拉 选 框 和 一 个 文本 框 ， 分 别 用 来 输入 记录 描述 、 消 费 类 型 和 消费 金额 。 页 面 如 图 10-21 
所 示 。 


添加 一 条 消费 记录 


图 10-21 添加 消费 记录 


为 了 通过 这 个 页 面 提交 消费 记录 ， 我 们 可 以 在 判断 用 户 提交 了 表单 之 后 ， 获 取 并 保存 
用 户 填写 的 数据 到 相应 的 变量 ， 然 后 将 这 些 变量 的 值 存 入 数据 库 的 相应 数据 表 中 。 


汪汪 
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按照 这 个 思路 ， 在 这 个 页 面 的 </body> 标 签 前 添加 如 下 PHP 脚本 : 


<?php 

// 判 断 用 户 是 否 提交 了 表单 

if(isset($ REQUEST['addExpense'])){ 
// 若 用 户 提交 了 表单 ， 则 获取 用 户 提 交 的 数据 
SeDesc = $ REQUEST['desc']; 
S$eCategory = $ REQUEST['category']; 
$eAmount = $ REQUEST['amount']; 
$userId = $ SESSION['userId']; 


Smsg = ""'; 


// 建 立 bookKeeping 类 的 实例 ， 用 于 添加 消费 记录 


$bk = new bookKeeping() : 


Smsg = $bk ->AddSingleExpense ($eDesc, $eAmount, $eCategory, $userId); 


echo '<script language="javascript">document .getElementById ("msg"). 


innerHTML="<p>' .$msg.'</p>"</script>"'; 


2 


在 上 面 这 段 脚 本 中 ， 我 们 首先 判断 用 户 是 否 提交 了 表单 。 若 用 户 提交 了 表单 ， 则 获取 
用 户 提 交 的 数据 ， 并 创建 一 个 bookKeeping 类 的 实例 ， 并 使 用 其 AddSingleExpense() 方 法 ， 


将 用 户 提交 的 数据 存 入 数据 库 中 。 


需要 注意 的 是 ，AddSingleExpense() 方 法 自 带 了 数据 验证 的 脚本 ,所 以 在 这 个 页 面 上 我 


们 就 没有 验证 用 户 输入 的 数据 。 
2. 为 “添加 收入 记录 ”页 面 添加 功能 代码 


为 添加 收入 记录 的 页 面 添加 功能 的 思路 与 为 添加 消费 记录 的 页 面 添加 功能 的 思路 是 
一 样 的 。 都 是 通过 获取 用 户 输入 的 数据 并 将 其 存 入 到 数据 库 中 。 页 面 如 图 10-22 所 示 。 


欢迎 您 , tobunka 


适 加 并 要 记录 A 
EN 
撕 还 : 
牙 量 导入 记录 
二 DR 
类 到 :工资 [=] 
全: 
EE D5 


图 10-22 添加 收入 记录 


为 了 实现 预定 的 功能 ， 其 脚本 如 下 : 


<?php 


“5 
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// 判 断 用 户 是 否 提交 了 表单 


El 


} 
2 


3. 为“ 


isset($ REQUEST['addIncome']) )1{ 


// 若 用 户 提交 了 表单 ， 则 获取 用 户 提交 的 数据 
SiDpesc = $ REQUEST['desc']; 
SiCategory = $ REQUEST['category']; 
$iAmount = $ REQUEST["'amount']; 
SuserId = $ SESSION['userId']; 


Smsg = "7 


// 建 立 bookKeeping 类 的 实例 ， 用 于 添加 收入 记录 
$bk = new bookKeeping(); 


Smsg = $bk -> AddSingleIncome ($iDesc, $iAmount, $iCategory, $userId); 


echo '<script language="javascript">document .getElementById ("msg"). 
innerHTML="<p>"' .$msg.'</p>"</script>"'; 


批量 上 传导 入 记录 ”页 面 添加 功能 


批量 上 传 可 以 说 是 这 个 记 账 工具 中 比较 有 特色 的 一 个 功能 。 使 用 批量 上 传 ， 可 以 帮助 
用 户 使 用 规定 的 标记 语言 事先 在 文本 文件 中 录入 消费 和 收入 记录 ， 然 后 再 一 次 性 的 导入 到 
数据 库 中 。 这 个 页 面 如 图 10-23 所 示 。 


在 批量 :| 


<?php 


欢迎 您 ,tobunka | 


才 加 消 要 记 叶 批量 添加 记录 

活 j0 必 入 记录 选 笃 上 传 文件 [Bowse- ] 
批量 导入 记录 

二 看 3g 才 Ei 


图 10-23 ”批量 添加 记录 


上 传记 录 的 页 面 上 ， 我 们 在 “</body> ”标签 前 添加 如 下 的 PHP 脚本 : 


// 检 查 用 户 是 否 点 击 了 “开始 上 传 ” 按 钮 。 若 是 ， 则 开始 获取 与 上 传 文件 相关 的 信息 


3 


isset($ REQUEST['batchUpload'])){ 
Stemp name = $ FILES['user file']['tmp name']; 
Sname = $ FILES['user file']['name']; 
$typeof = $ FILES['user file']['type']; 
$userId = $ SESSION['userId']; 


$msg = 2 


全 全 全 全 人生 全 全 们 全 全 全 人 全 位 输出 错误 提示 


if($temp name == "" 


人 
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!preg match ('" /text/' ,Stypeof) ){ 

Smsg = "上 传 失败 或 格式 不 正确 。"' ; 

} else { 
/ /否则 ， 指 定 用 于 保存 用 户 上 传 文件 的 文件 夹 路 径 
$destination = dirname( FILE ).'\upload\\' .$name; 
// 移 动 临时 文件 至 指定 的 该 文件 夹 中 
move uploaded file (Stemp name, $destination); 
// 新 建 名 为 $bk 的 bookKeeping 类 对 象 
$bk = new bookKeeping (); 
// 使 用 $bk 对 象 的 readTemp 方法 处 理 用 户 上 传 文件 中 的 数据 
Smsg = $bk->readTemp ($name, $userId); 


h 
// 向 id 为 “msg” 的 div 标签 中 输出 信息 。 
echo "<script language="javascript">document .getElementById ("msg") . 
innerHTML= 
"<p>' .$msg.'</p>"</script>'; 
1 
> 


然后 使 用 定义 好 的 用 户 名 和 密码 登录 记 账 工具 ， 单 击 页 面 左 侧 导 航 栏 中 的 “批量 导入 
记录 ”链接 ， 进 入 “批量 添加 记录 ”页 面 。 然 后 单 击 “浏览 ” 按 钮 找到 需要 上 传 的 文件 。 
单 击 “ 开 始 上 传 ”按钮 ， 随 后 ， 在 按钮 的 下 方 会 出 现 处 理 结果 ， 如 图 10-24 所 示 。 


| 


€ 9 CT EI CIE SY 


俩 Fovortes 眉 Add Epense 稻 ~ 国 ~- 可观 ~ Page- sfcy- Toos> 且 -| 


和 园 源 上 人 文件 


Er 


号 天 可 了 2 条 尊 要 沁 录 和 1 称 梭 入 元 好. 


图 10-24 成功 添加 了 2 条 消费 记录 和 1 条 收入 记录 
随后 可 以 通过 phpMyAdmin 查看 数据 库 中 是 否 出 现 了 这 三 条 属于 名 为 “user” 用 户 的 
至 此 ， 我 们 已 经 完成 了 记 账 工具 的 批量 导入 功能 。 剩 余 的 几 个 页 面 ， 内 容 大 同 小 异 ， 
读者 可 以 尝试 着 自己 完成 。 


4. 为 “查看 记录 列表 ”页 面 添加 功能 
在 这 个 页 面 里 ， 用 户 可 以 查询 指定 时 间 范 围 内 的 消费 和 收入 记录 ， 并 对 这 些 记录 进行 
修改 或 删除 其 中 的 若干 条 记录 。 页 面 如 图 10-25 所 示 。 
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er 门 
再 J 应 要 旭 查看 记录 列表 
渗 j0 改 入 记录 [013 回 年 [10 国 | 月 及 以前 的 [ 消 责 记录 思 ] [ 开 瓜 喜 找 | 
批量 入 记录 
E53 
图 10-25 查看 记录 列表 
在 查询 到 记录 时 ， 页 面 如 图 10-26 所 示 。 
Be ,user 

ET ee 
和 AE 竹 2013 回 年 人 0 加 ] 请 J 和 i 的 | 消 圳 记录 吕 ] [ 开 圾 喜 损 ] 
地 县 AE 归 

总 共 找 到 1 条 消费 志 景 

EE EE 
局 汪 大 一 全 | 520.00‖ 语 品 [| 


图 10-26 ”查询 到 记录 时 的 “查询 列表 页 面 ” 


在 查询 到 记录 时 ， 我 们 可 以 对 查询 到 的 记录 做 修改 或 删除 的 操作 ， 为 此 还 需要 创建 两 
个 新 页 面 分 别 用 于 修改 用 户 指定 的 记录 和 删除 用 户 指定 的 记录 。 

关于 这 两 个 页 面 的 HTML 部 分 ， 可 以 参考 我 们 随 书 提供 的 源 代 码 。 这 里 重点 来 看 看 如 
何 使 用 PHP 实现 查询 、 修 改 和 删除 。 

(1) 查询 指定 时 间 前 的 消费 和 收入 记录 

为 了 实现 查询 指定 时 间 前 的 消费 和 收入 记录 , 需要 在 查询 页 面 中 ID 为 table 的 DIV 中 
添加 如 下 代码 : 


<?php 
if(isset($ REQUEST['search']))t{ 
// 获 取 用 户 提交 的 数据 
$setYear = $ REQUEST['year']; 
$setMonth = $ REQUEST['month']; 
SrecordType = $ REQUEST['type']; 
$userId = $ SESSION['userId']; 


ss 
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// 创 建 bookKeeping 类 实例 ， 并 使 用 该 实例 的 SearchForRecords 方法 查找 记录 
Sbk = new bookKeeping (); 


S$records = $bk -> SearchForRecords ($setYear, $setMonth, 
S$recordType, $userId); 


// 输 出 查询 到 的 记录 到 表格 中 
if($recordType == 'EXP'){ 
echo '<p> 总 共 找到 ' .count ($records) . ' 条 消费 记录 </p>"'; 
echo '<table><tr><td> 记 录 号 </td> 
<td> 消 费 描述 </td> 
<td> 消 费 金额 </td> 
<td> 消 费 类 型 </td> 
<td> 消 费 日 期 </td> 
<td> 操 作 </tq></tr>'; 


if(count($records) > 0) { 
foreach ($records as $key => $value) { 
echo '<tr><td>EXP' .sprintf('%05s',$value[0]).'</td> 
<td>"' .$value[1].'</td> 
<td>"' .$value[2].'</td> 
<td>"' .$value[3].'</td> 
<td>' .$value[4].'</td> 
<td> 
<a href="08.6 Book Keeping Modify.php 
?t=expense&id=' .$value[0] .'"> 修 改 </a> | 
<a href="08.7 Book Keeping Remove.php 
?t=expense&id=' .$value[0] .'"> 删 除 </a> 
> 
} 
} else { 
echo '<tr><td colspan="6"> 未 找到 相应 消费 记录 。</td></tr>'; 
} 


echo '</table>'; 
} else { 
echo '<p> 总 共 找 到 ' .count ($records) .' 条 收入 记录 </p>'; 
echo '<table><tr><td> 记 录 号 </td> 
<td> 收 入 描述 </td> 
<td> 收 入 金额 </td> 
<td> 收 入 类 型 </td> 
<td> 收 入 日 期 </td> 
<td> 操 作 </td></tr>'; 


if(count($records) > 0) { 
foreach ($records as $key => $value) { 
echo '<tr><td>INC'.sprintf('%05s',$value[0]).'</td> 

RE HvaAlualll <A/Ead> 

<td>' .$value[2].'</td> 

<td>” .Svaluel3l. "</o> 

REG> .Hvalaetlal. "</Ed> 

<td> 
<a href="08.6 Book Keeping Modify.php 
2t=income&gid=' .$value[0] .' "> 修改 </a> | 
<a href="08.7 Book Keeping Remove.php 
2t=income&gid=' .$value[0] .' "> 删除 </a> 

</td></tr> "> 


= 
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} else { 
echo '<tr><td colspan="6"> 未 找到 相应 收入 记录 。</td></tr>'; 
1; 


echo '</table>'; 


} 

2> 

在 上 面 这 段 代码 中 ， 我 们 主要 做 了 三 件 事 。 第 一 件 是 判断 用 户 是 否 提交 了 表单 ， 并 获 
取 用 户 提交 的 数据 。 第 二 件 是 创建 bookKeeping 类 实例 , 并 使 用 该 实例 的 SearchForRecords 
方法 来 查找 用 户 指定 时 间 之 前 的 指定 类 型 的 记录 ,并 将 查询 到 的 内 容 存 入 数组 Srecords 中 。 
第 三 件 是 根据 用 户 指定 的 记录 类 型 ， 遍 历数 组 $records， 并 生成 表格 。 

在 这 三 件 事 中 ， 第 一 件 事情 与 在 其 他 页 面 上 的 判断 类 似 ， 第 二 件 事 情 ， 应 该 在 为 页 面 
添加 功能 前 的 准备 工作 中 已 经 完成 了 。 因 此 ， 第 三 件 事情 就 是 此 脚本 的 重点 。 由 于 此 处 只 
是 巩固 我 们 之 前 学 习 到 的 内 容 ， 所 以 把 HTML 代码 和 PHP 脚本 故 合 在 了 一 起 。 在 实际 开 
发 的 最 佳 实践 中 ， 我 们 应 该 把 呈现 层 〈( 即 HTML 代码 ) 和 控制 层 ( 即 PHP 脚本 ) 分 离开 
来 ， 以 便 将 来 的 脚本 维护 工作 。 

在 根据 用 户 指定 的 时 间 范 围 和 记录 类 型 查找 到 相关 记录 后 ， 我 们 可 以 通过 每 条 记录 的 
操作 栏 中 提供 的 “修改 ”和 “删除 ”链接 来 修改 和 删除 该 记录 。 

(2) 修改 指定 的 消费 和 收入 记录 

为 了 实现 修改 指定 消费 和 收入 记录 ， 我 们 需要 使 用 bookKeeping 类 的 
ModifyExpenseEntry 和 ModifyIncomeEntry 两 个 方法 。 这 两 个 方法 都 要 求 用 户 提供 四 个 参 
数 : 前 者 要 求 的 参数 有 消费 记录 编号 (expenseId) 、 新 消费 描述 (eDesc) 、 新 消费 金额 
(eAmount) 及 新 消费 类 型 (eCategory) ; 而 后 者 要 求 的 参数 为 收入 记录 编号 (incomeId) 、 
新 收入 描述 〈iDesc) 、 新 收入 金额 (iAmount) 及 新 收入 类 型 (iCategory) 。 

另外 ， 在 查询 页 面 中 ， 用 户 既 可 以 查询 消费 记录 ， 也 可 以 查询 收入 记录 ， 在 修改 记录 
的 时 候 ， 我 们 也 需要 通知 修改 页 面 需要 修改 记录 到 底 是 消费 记录 还 是 收入 记录 。 因 此 ， 需 
要 使 用 查询 页 面 中 的 “修改 ”链接 的 URL 向 修改 记录 的 页 面 传递 “记录 类 型 ”和 “记录 编 
号 ”两 个 参数 ， 然 后 在 修改 记录 的 页 面 中 根据 传递 过 来 的 “记录 类 型 ”和 “记录 编号 ” 查 
找 相 应 记录 。 

综 上 所 述 , 我 们 需要 在 修改 记录 的 页 面 上 完成 三 件 事 : 第 一 件 事 是 获取 用 户 通过 URL 
传递 过 来 的 “记录 类 型 ” 和 “记录 编号 ”; 第 二 件 事情 是 根据 这 两 个 参数 查询 记录 的 详情 ， 
并 将 详情 各 项 填充 到 用 户 修改 记录 时 使 用 的 表单 中 ;第 三 件 事情 是 处 理 用 户 更 新 的 数据 。 
为 此 ， 我 们 需要 在 修改 记录 的 页 面 的 不 同位 置 插 入 PHP 脚本 。 具 体 如 下 : 

为 了 完成 第 一 件 事 ， 我 们 需要 在 页 面 的 头 部 插入 如 下 PHP 脚本 : 


if(isset($ SESSION['user'])) { 
$recordType = $ REQUEST['t"']; 
$recordId = $ REQUEST['id']; 


$title = ($recordType == "expense") ? ' 消 费 记录 ' : ' 收 入 记录 '; 
$bk = new bookKeeping(); 


$details = $bk -> GetDetails($recordType, S$recordId); 


Rs 
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在 这 段 脚 本 中 ， 我们 使 用 $_REQUEST 数组 获取 了 用 户 通 过 URL 传递 过 来 的 数据 。 然 
后 创建 了 一 个 bookKeeping 类 实例 ， 并 使 用 该 实例 的 GetDetails 方法 获取 了 用 户 指定 记录 
的 详情 。 

为 了 完成 第 二 件 事 ， 我 们 将 获取 到 的 详情 填充 到 了 用 户 在 修改 记录 时 需要 用 到 的 表单 
中 。 这 一 部 分 粹 合 HTML 代码 和 PHP 脚本 。 看 上 去 虽然 复杂 ， 但 其 他 原理 很 简单 ， 那 就 
是 使 用 echo 方法 将 获取 到 的 详情 各 项 “输出 ”到 对 应 的 表单 元 素 的 “Value” 属 性 中 。 具 
体 脚本 可 以 参考 我 们 随 书 提供 的 源 代码 。 

为 了 完成 第 三 件 事 ， 我 们 将 在 修改 记录 的 页 面 的 </body> 标 签 前 添加 如 下 PHP 脚本 : 

<?php 

// 判 断 用 户 是 否 提交 了 表单 


if(isset($ REQOUEST['modRecord']))1{ 
// 若 用 户 提交 了 表单 ， 则 获取 用 户 通 过 表单 和 URL 地 址 传递 的 所 有 参数 
SrecordType = $ REQUEST['t'"]7 
SrecordId = $ REQUEST['id']7 
$modDesc = $ REQUEST['desc']; 
SmodCategory = $ REQUEST['category']; 
$modAmount = $ REQUEST['amount']; 


// 创 建 一 个 bookKeeping 类 的 实例 ， 并 根据 变量 SrecordType 的 值 执行 不 同 的 修改 方法 
$bk = new bookKeeping (); 
if($recordType == 'expense'){ 
$msg = $bk->ModifyExpenseEntry($recordId, $modDesc, 
$modAmount, $modCategory); 
} else { 
$msg = $bk->ModifyIncomeEntry($recordId, $modDesc, 
S$modAmount, $modCategory); 
| 


// 将 修改 结果 返回 到 页 面 中 名 为 msg 的 DIV 元 素 中 


echo "<script language="javascript">document .getElementById ("msg") . 
innerHTML="<p>"' .$msg.'</p>"</script>"'; 

> : 

这 段 脚 本 首先 判断 用 户 是 否 提供 了 表单 。 若 用 户 提交 了 表单 ， 则 获取 用 户 通过 表单 和 
URL 地 址 传递 的 所 有 参数 。 然 后 创建 一 个 bookKeeping 实例 ， 并 使 用 该 实例 的 
ModifyExpenseEntry 方法 或 ModifyIncomeEntry 方法 将 获取 到 的 参数 更 新 到 数据 库 中 的 指 

(3) 删除 指定 的 消费 和 收入 记录 

为 了 实现 删除 指定 的 消费 和 收入 记录 ， 我 们 需要 使 用 bookKeeping 类 的 
DeleteExpenseEntry 和 DeletemcomeEntry 方法 。 这 两 个 方法 均 只 要 求 提供 一 个 参数 ， 即 用 
户 指定 记录 的 记录 编号 。 这 也 就 是 说 , 一 旦 执行 这 两 个 方法 删除 指定 的 记录 且 记 录 存 在 时 ， 
记录 就 会 直接 被 删除 。 

为 了 提供 给 用 户 一 个 后 悔 的 机 会 ， 我 们 需要 在 删除 记录 的 页 面 上 添加 一 个 “确认 ”机 
制 ， 在 用 户 单 击 记录 列表 中 的 “删除 ”按钮 来 到 删除 页 面 时 ， 首 先 会 看 到 一 个 确认 框 ( 如 
图 10-27 所 示 ) ， 用 户 可 以 在 这 个 确认 框 中 看 到 其 指定 的 记录 的 详细 内 容 。 如 果 确 认 需 要 
删除 这 个 记录 ， 则 可 以 单 击 确认 框 下 方 的 “删除 ”链接 删除 该 记录 。 反 之 ， 则 可 以 单 击 确 
认 框 下 方 的 “取消 ”链接 返回 查询 页 面 。 
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图 10-27 删除 记录 前 的 确认 机 制 
为 了 实现 这 个 功能 ， 我 们 需要 在 页 面 的 头 部 添加 如 下 PHP 脚本 : 
// 判 断 用 户 是 否 为 已 登录 用 户 
if(isset($ SESSION['userId']))1{ 
// 若 为 已 登录 用 户 ， 则 获取 用 户 通过 URL 传递 到 本 页 面 的 参数 


$recordType = $ REQUEST['t"]7 
$recordId = $ REQUEST['id']; 


$title = ($recordType == 'expense') ? "消费 记录 "” : "收入 记录 "; 


// 新 建 一 个 bookKeeping 实例 
$bk = new bookKeeping (); 


// 判 断 用 户 是 否 确认 删除 指定 的 记录 
if (isset($ REQUEST['confirm'])){ 
// 根 据 用 户 提交 的 记录 类 型 ， 来 执行 bookKeeping 实例 的 方法 


if ($recordT == 'expense'){ 
Smsg = $bk->DeleteExpenseEntry ($recordId); 
} else { 


$msg = $bk->DeleteIncomeEntry ($recordId); 


} else { 
Srecord = $bk->GetDetails($recordType, S$recordId); 


// 判 断 指定 记录 是 否 存在 。 由 于 GetDetails 方法 返回 值 为 数组 ， 这 里 只 需要 判断 变量 
Srecord 元 素 个 数 是 否 为 0 


if(count (Srecord) > 0){ 


// 若 数组 元 素 个 数 不 为 0， 则 开始 输出 获取 到 的 信息 


if($recordType == "expense')1{ 
Smsg = '<p> 确 定 要 删除 如 下 消费 记录 吗 ?' .'</p>'; 
Smsg .= '<ul><1i>[ 记 录 号 ] EXP' .sprintf('%05s',$recordId). 
re/ 


Smsg .= "<1i>[ 消 费 描述 ] 
$msg -= '<1i>[ 消 费 金额 ] 
Smsg .= "<1i>[ 消 费 类 型 ] 
Smsg -= "<1i>[ 消 费 日 期 ] 


”Srecordl0Ol "</li> 
ECGordhllI <A 
Eeeeordl I /li 
ecordlal </ALiS</ul> Ys 
} else { 

$msg = '<p> 确 定 要 删除 如 下 收入 记录 吗 ? </p>'; 
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} 


Smsg - 


A 


Smsg - 
$msg . 
Smsg . 
Smsg . 


[i ee ee 


"<1i>[ 收 入 描述 ] 
"<1i> [收入 金额 ] 
"<1i> [收入 类 型 ] 
"<1i> [收入 日 期 ] 


"<ul><1i>[ 记 录 号 ] EXP' .sprintf('%05s',$recordId). 


"mocord[lOl > 
ell ls eA 
mredl 2 Ce 
OreoOrdl3l cA/ 


在 上 面 这 段 脚 本 中 ,我 们 获取 了 用 户 通 过 URL 传递 的 参数 , 然后 创建 一 个 bookKeeping 
类 的 实例 并 使 用 其 GetDetails 方法 来 获取 用 户 指定 记录 的 详情 ， 然 后 将 详情 存 入 变量 Smsg 
中 。 若 用 户 确 认 要 删除 显示 的 记录 ， 则 将 删除 记录 的 方法 返回 的 消息 存 入 变量 Smsg 中 。 

接着 我 们 在 页 面 中 ID 为 “confirm-box” 的 DIV 中 ， 插 入 如 下 脚本 : 


<?php echo $msg; ?> 


<p id= 


"right-align"> 


<?php if (isset($ REQUEST['confirm'])) { ?> 

<a href="08.2 Book Keeping AddExpense.php"> 返 回首 页 </a> 
<?php } else { ?> 
<a href="<?php echo '?t="'.$recordType.'&id=".$recordId.'gconfirm'; ?>"> 
删除 </a> 
<a href="08.5 Book Keeping Query.php"> 取 消 </a> 
<?php } ?> 


</p> 


在 这 段 脚本 中 , 我 们 输出 了 变量 Smsg 的 值 , 并 根据 用 户 是 否 已 确认 删除 显示 的 记录 来 
决定 确认 框 下 方 显示 的 链接 。 若 用 户 处 于 确认 阶段 则 确认 框 下 方 的 链接 为 “删除 ”和 “ 取 
消 ”。 单 击 “ 删 除 ” 可 删除 显示 的 记录 ， 单 击 取消 可 以 回 退 到 记录 查询 页 面 。 若 用 户 已 经 
确认 删除 显示 的 记录 ， 则 确认 框 正文 的 链接 为 “返回 首页 ”。 单 击 该 链接 可 以 回 到 “添加 
消费 记录 ”页 面 。 


(1) 使 
(2) 使 
(3) 使 
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题 


用 文本 文件 做 为 存储 介质 ， 编 写 一 个 日 记 本 应 用 。 
用 XML 文件 做 为 存储 介质 ， 编 写 一 个 图 库 应 用 。 
用 MySQL 数据 库 为 数据 存储 介质 ， 编 写 一 个 用 户 信息 管理 系统 。 
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PHP 为 我 们 提供 了 大 多 数 编程 语言 有 的 文件 操控 功能 。 使 用 PHP， 可 以 很 方便 获取 操 
作 系 统 及 存储 在 操作 系统 中 的 文件 信息 。 具 体 说 来 , 我 们 可 以 使 用 PHP 在 操作 系统 中 创建 、 
复制 、 删 除 、 查 找 和 移动 文件 ， 也 可 以 使 用 PHP 运行 操作 系统 中 的 任何 程序 。 同 样 ， 我 们 
还 可 以 在 相互 连接 的 两 台电 脑 间 通过 PHP 或 电子 邮件 传送 消息 和 文件 。 

在 本 章 中 就 来 看 看 如 何 使 用 PHP 来 实现 这 一 切 。 


11.1 管理 文件 


信息 都 是 以 文件 的 形式 存储 在 硬盘 上 的 。 但 是 为 了 便于 管理 ， 我 们 把 这 些 文件 分 门 别 
类 地 放 入 不 同 的 文件 夹 内 。 由 文件 和 文件 夹 组 成 的 系统 称 为 文件 系统 。 通 常 来 说 ， 一 个 文 
件 系统 的 结构 是 分 层 的 。 在 这 个 分 层 结构 的 顶端 具有 一 个 文件 夹 ， 称 为 根 文件 夹 。 在 
Windows 系统 中 根 文 件 夹 为 “C:\”， 而 在 Linux 系统 中 根 文件 夹 为 “/”。 所 有 的 文件 和 文 
件 夹 都 存放 在 根 文件 夹 中 。 文 件 夹 中 可 以 包含 文件 和 子 文件 夹 。 理 论 上 来 说 ， 一 个 文件 夹 
可 以 包含 的 文件 夹层 级 是 无 限 的 。 

使 用 文件 夹 来 管理 文件 是 一 件 很 逻辑 的 事情 。 然 而 ， 对 于 实现 文件 夹 的 方式 来 说 ， 其 
本 身 其 实 也 是 一 种 文件 。 它 是 一 份 关 于 存储 在 该 文件 夹 中 的 文件 和 子 文件 夹 的 列表 。 通 过 
该 列表 ， 操 作 系 统 可 以 很 方便 地 找到 相应 的 文件 和 子 文件 夹 。 

正如 在 上 一 章 里 学 到 的 , PHP 为 我 们 提供 了 读 取 、 修 改 和 删除 文件 的 方法 。 除 此 之 外 ， 
还 可 以 使 用 PHP 对 文件 和 文件 夹 进行 检查 、 删 除 、 复 制 和 重 命名 等 操作 。 在 本 章 里 ， 我 们 
会 涉及 到 几乎 所 有 与 文件 管理 相关 的 方法 , 但 是 PHP 为 我 们 提供 的 方法 却 不 仅 限于 此 。 如 
果 想 了 解 更 多 ,可 以 在 PHP 官方 网 站 上 的 手册 中 查找 。 如 果 找 不 到 想 要 的 功能 ， 而 该 功能 
可 以 通过 系统 中 的 其 他 某 些 应 用 程序 或 命令 来 实现 ， 就 可 以 通过 调用 操作 系统 命令 来 完成 
相应 的 任务 。 


11.1.1 获取 文件 信息 


我 们 需要 了 解 某 些 文件 的 相关 信息 。 比 如 ， 对 于 一 张 图 片 来 说 ， 可 能 需要 知道 这 张 图 
片 有 多 大 、 尺 寸 是 多 少 、 什 么 时 间 以 什么 格式 存放 在 系统 中 。 如 果 这 张 图 片 是 一 张 数码 
相机 拍摄 的 像 片 ， 还 需要 了 解 这 张 照 片 是 用 什么 设备 拍摄 的 、 光 圈 是 多 少 等 更 加 详细 的 
信息 。 

我 们 可 以 使 用 角 e_exists0 方 法 来 判断 某 个 文件 是 否 存 在 ， 该 方法 返回 一 个 逻辑 值 。 
比如 : 
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$result = file exists("stuff.txt'")7 
if (!Sresult) 


echo "file not found!"7 

} 

当 我 们 得 知 文件 存在 时 ,就 可 以 查看 其 中 的 信息 了 。 表 11-1 列 出 了 常用 的 与 文件 管理 

相关 的 函数 和 方法 。 
表 11-1 常用 的 文件 管理 函数 
模式 说 明 输 出 

判断 指定 的 对 象 是 否 为 一 个 文件 , 而 不 是 四 
is_file("stuff.txt") 文件 夹 或 其 他 特殊 格式 的 文件 TRUE 或 者 FALSE 
is_dir("stuff .txt") 判断 指定 的 对 象 是 否 为 一 个 文件 夹 TRUE 或 者 FALSE 
is_executable("stufftxt") | 判断 指定 的 对 象 是 否 为 一 个 可 执行 文件 | TRUE 或 者 FALSE 


判断 指定 的 对 象 是 否 可 写 
判断 指定 的 对 象 是 否 可 读 


is_writable("stuff.txt") 
is_readable("stuff.txt") 


fileatime("stuff.txt") 返回 最 近 一 次 访问 指定 文件 的 时 间 
filectime("stuff.txt") 返回 创建 该 文件 的 时 间 
filemtime("stuff.txt") 返回 最 近 一 次 修改 指定 文件 的 时 间 
filegroup("stuff.txt") 返回 指定 文件 所 属 组 的 ID 


返回 指定 文件 的 所 有 者 ID 
返回 指定 文件 的 大 小 ， 单 位 为 字 节 


fileowner("stuff.txt") 
filesize("stuff.txt") 


filetype("stuff.txt)" 返回 指定 文件 的 类 型 


返回 指定 路 径 中 的 文件 名 
返回 指定 路 径 中 的 路 径 


basename("/dir/stuff.txt") 
dimame("/dir/stuff.txt") 


TRUE 或 者 FALSE 

TRUE 或 者 FALSE 

UNIX 时 间 戳 或 FALSE 

UNIX 时 间 戳 或 FALSE 

UNIX 时 间 惟 或 FALSE 

代表 文件 组 ID 的 整数 或 FALSE 
代表 文件 所 有 者 ID 的 整数 或 
FALSE 

代表 文件 大 小 的 整数 或 FALSE 
代表 文件 类 型 的 字符 串 (取舍 可 
能 为 file、 dir、 file、 link 和 char) 
或 FALSE 

stuff.txt 

/dir 


除了 上 面 这 些 函 数 之 外 , 我 们 还 可 以 使 用 pathinfo0 来 获取 一 个 描述 文件 在 系统 中 的 位 


置 的 数组 。 比 如 : 
Spinfo = pathinfo("/dir/stuff.txt"); 


echo $pinfo[dirname]; // 输 出 “/dir” 
echo S$pinfo[basename]; 
echo S$pinfo[extension]; 


// 输 出 “tzxt” 


11.1.2 ”复制 、 重 命名 和 删除 文件 


// 输 出 “stuff.txt” 


在 上 一 章 里 ， 我 们 知道 了 如 何 创建 一 个 文本 文件 、 如 何 向 文件 中 写 入 信息 。 在 本 小 节 


来 了 解 一 下 如 何 复制 、 重 命名 和 删除 文件 。 


需要 注意 的 是 , 在 使 用 copy0 函 数 进行 文件 复制 操作 时 , 我 们 只 能 复制 已 存在 的 文件 。 
在 复制 完成 后 ， 会 得 到 两 个 内 容 一 样 、 文 件 名 不 一 样 的 文件 。 通 常 在 需要 备份 某 个 文件 时 
使 用 复制 操作 。 我 们 可 以 使 用 类 似 如 下 的 脚本 复制 一 个 指定 文件 的 内 容 到 另 一 个 文件 中 : 


copy ("fileold.txt","filenew.txt"); 


在 PHP 执行 该 语句 后 ，fileold.txt 这 个 已 存在 于 操作 系统 中 的 文件 会 被 复制 为 
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filenew.txt 这 个 文件 。 在 执行 过 程 中 ，PHP 会 磁 到 两 种 情况 : 

口 若 fenew.txt 文件 不 存在 ， 则 PHP 会 主动 创建 filenew.txt 文件 并 将 fleold.txt 文件 

的 内 容 复制 到 flenew.txt 中 。 

口 若 filenew.txt 文件 已 存在 ， 则 PHP 会 用 fileold .txt 文件 的 内 容 来 覆盖 filenew.txt 文 
件 的 内 容 ， 也 就 是 说 ，filenew.txt 文件 原 有 的 内 容 会 消失 。 

若 不 想 覆 盖 某 个 文件 的 内 容 ， 可 以 使 用 锯 e_exits0 函 数 进行 判断 。 比 如 : 


if(!file exists("filenew.txt")){ 
copy ("fileold.txt","filenew.txt"); 
} else { 

echo "file already exists!"; 


| 
类 似 地 ， 我 们 可 以 使 用 rename0 函 数 来 重 命 名 一 个 指定 的 文件 ， 如 : 


rename ("oldname .txt", "newname .txt"); 

PHP 在 执行 该 语句 时 ， 会 进行 如 下 的 逻辑 判断 。 

口 第 一 步 ” 先 判断 oldname.txt 文件 是 否 存在 。 若 存在 ， 则 进行 第 二 步 。 若 不 存在 ， 
则 发 出 如 下 警告 : 


Warning: rename (oldname .txt,newname .txt) : The system cannot find the file 
specified. (code: 2) in C:\greatwal1\chapterl11\Ex11-1.php on line 2 


口 第 二 步 ” 判 断 newname.txt 文 件 是 否 存 在 。 若 存在 ， 则 发 出 如 下 警告 ， 若 不 存在 ， 
则 将 oldname.txt 的 文件 名 替换 为 newname.txt: 


Warning: rename (oldname .txt,newname .txt) : File exists in c:test.php on line 2. 


类 似 地 ， 如 果 想 移 除 某 个 文件 ， 可 以 使 用 unlink0 方 法 ， 如 : 

unlink("badfile.txt"); 

PHP 在 执行 该 语句 时 , 会 判断 指定 文件 是 否 存在 。 如 果 存 在 则 删除 该 文件 ， 若 不 存在 ， 
则 报错 。 

现在 来 看 一 段 脚本 。 在 下 面 这 段 脚本 中 ， 我 们 使 用 fopen0 消 数 、copy0 函 数 、rename() 
函数 和 unlink() 函 数 。 

【 例 11.1】 复制 、 重 命名 和 删除 文件 。 


和 <?php 
2 fopen('abc.txt', 'w'); // 创 建文 件 abc .txt 
4 if(!file exists('abc.txt')) // 判 断 文件 abc.txt 是 否 存在 
5 [ 
6 echo "the specified file does not exist."; 

// 若 不 存在 ， 输 出 “指定 文件 不 存在 ”的 提示 
世 } else { 
8 if (!file exists ("new abc.txt"))// 判 断 文件 new abc.txt 是 否 存在 
9 1 
10 copy ('abc.txt'，'new abc.txt'); 

// 若 不 存在 ， 则 将 文件 abc .txt 复制 到 new_abc .txt 中 
了 } else { 
下 2 echo 'The target file already exists. The COPY is terminated.'; 
3 } 
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14 
Te if (!file _ exists('abc new.txt')) // 判 断 文件 abc new.txt 是 否 存 在 
16 { 
Ei rename ('new abc.txt', 'abc new.txt'); 
// 若 不 存在 ， 则 将 文件 重 命名 为 new_abc .txt 
18 } else { 
19 echo "There is already a file with the new name specified. 
The RENAME is terminated"; 
20 } 
思科 } 
人 2 
23 if (file exists('abc.txt')) // 判 断 文件 abc .txt 是 否 存在 
24 { 
25 unlink('abc.txt'); // 若 存在 ， 则 删除 文件 abc .txt 
26 } else { 
7 echo "The file does not exist. The DELETION is terminated."; 
28 } 
2 2> 


在 上 面 这 段 脚本 中 ， 我 们 首先 使 用 fopen0 函 数 创 建 了 一 个 可 读 写 的 文本 文件 abc.txt。 
然后 使 用 file_exists0 〇 函数 来 判断 文件 abc.txt 是 否 创建 。 如 果 文件 已 创建 ， 则 使 用 copyO 函 
数 将 其 复制 一 份 至 new_abc.txt， 再 使 用 rename() 函 数 将 文件 new_abc.txt 重 命 名 为 
abc_new.txt。 最 后 ， 使 用 unlink0 函 数 删 除 最 先 创建 的 abc.txt 文件 。 

运行 该 脚本 后 ， 浏 览 器 里 不 会 出 现任 何 信 息 ， 而 当前 文件 夹 下 则 多 出 了 一 个 名 为 
abc_new.txt 的 文本 文件 。 


11.1.3 组织 文件 


在 操作 系统 中 ， 文 件 是 通过 文件 夹 进行 整理 的 。 在 本 小 节 里 ， 我 们 就 来 看 看 如 何 修建 
和 移 除 文件 夹 ， 以 及 如 何 获取 某 文件 夹 中 的 文件 和 文件 夹 列表 。 

如 果 需 要 创建 一 个 文件 夹 ， 我 们 可 以 使 用 mkdir0 函 数 。 如 : 

mkdir("testdir"); 

该 语句 会 在 当前 路 径 下 创建 一 个 名 为 “testdir” 的 文件 夹 。 若 当前 路 径 下 已 存在 同名 
的 文件 夹 ， 则 PHP 会 显示 如 下 告警 : 


Warning: mkdir ("testdir"): File exists in c: on line 2. 


所 以 ， 在 创建 文件 前 ， 应 使 用 is_dir0 函 数 来 判断 一 下 同名 文件 夹 是 否 存 在 。 如 : 


if(!is dir("testdir")) 
L 
mkdir("testdir"); 
} else { 
echo "The specified directory already exists."; 


| 


如 果 需 要 在 某 个 文件 夹 下 创建 一 个 子 文件 夹 ， 可 以 使 用 绝对 路 径 ， 也 可 以 使 用 相对 路 
径 。 如 : 

mkdir("/root/nextdir/mynewdir"); // 绝 对 路 径 

mkdir("../mynewdir"); // 相 对 路 径 
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在 Windows 操作 系统 中 ， 所 谓 的 绝对 路 径 是 指 带 上 盘 符 的 路 径 ， 如 “C:\nextdir\ 
mynewdir”; 而 相对 路 径 则 是 指 以 脚本 文件 所 在 位 置 为 参照 点 的 路 径 。 若 脚本 文件 所 在 位 
置 的 绝对 路 径 为 “C:mextdir”， 而 我 们 需要 在 nextdir 文件 夹 下 创建 子 文件 夹 mynewdir， 
则 该 子 文件 夹 的 相对 路 径 为 “.\mynewdir”。 其 中 的 “..” 指 的 是 上 一 级 目录 ，“\” 后 的 
内 容 指 的 是 下 一 级 目录 。 

在 Linux 操作 系统 中 ， 不 存在 盘 符 的 说 法 ， 根 目录 为 root 目录 。 所 以 ， 绝 对 路 径 都 是 
以 root 目录 为 基准 的 ;而 相对 路 径 则 是 以 脚本 文件 所 在 的 位 置 为 基准 。 需 要 注意 的 是 ， 在 
Linux 操作 系统 中 ， 目 录 与 子 目 录 之 间 是 用 “/” 分 隔 的 ， 与 Windows 操作 系统 不 同 。 

如 果 需 要 从 当前 文件 夹 切 换 到 另 一 个 文件 夹 ， 则 可 以 使 用 chdir0 函 数 。 所 需 参 数 与 
mkdir0 相 同 ， 都 是 文件 夹 的 路 径 。 如 : 


chdir("..\mynewdir"); 


如 果 需 要 列 出 指定 文件 夹 中 的 所 有 文件 名 ， 可 以 先 用 opendir0 函 数 和 readdir0 函 数 将 
指定 文件 夹 读 取 到 一 个 变量 中 ， 接 着 使 用 while 循环 输出 指定 文件 夹 下 的 所 有 文件 名 。 其 
中 ，opendir0 函 数 可 以 看 作 是 一 个 游标 ， 它 返回 的 是 当前 游标 指针 指向 的 一 条 记录 ; 而 
readdir() 函 数 是 从 指针 指向 的 记录 读 取出 文件 名 。 

我 们 可 以 使 用 如 下 的 脚本 输出 上 一 章 的 例 程 目录 中 的 文件 名 列表 。 

【 例 11.2】 输出 指定 目录 下 的 文件 名 列表 。 


1 <?php 

2 

< $dh = opendir("..\chapter10"); // 将 指定 目录 读 取 到 一 个 游标 变量 中 
4 

5 $filename = readdir ($dh); 


// 将 游标 指针 指向 的 文件 的 文件 名 读 取 到 一 个 文本 变量 中 


6 
ys while($filename = readdir ($dh)) 
// 若 当前 游标 指向 不 是 列表 末尾 , 则 输出 该 文件 对 应 的 文件 名 
8 
四 echo $filename."<br>"; 
10 有 
> 


11.2 调用 操作 系统 命令 


在 Windows 操作 系统 中 , 我 们 可 以 通过 命令 行 窗口 向 操作 系统 发 出 各 种 各 样 的 指令 来 
完成 相应 的 操作 。 如 果 我 们 想 查看 某 个 文件 夹 中 的 文件 列表 ， 可 以 使 用 dir 命令 ;如 果 想 
要 复制 某 个 文件 到 指定 的 位 置 , 可 以 使 用 copy 命令 ; 如 果 想 要 创建 文件 夹 , 可 以 使 用 mkdir 
命令 。 

在 本 节 中 , 我 们 将 介绍 如 何 使 用 PHP 来 调用 操作 系统 的 命令 完成 相应 的 操作 。 完 成 本 
节 的 内 容 需 要 对 Windows 操作 系统 的 MS-DOS 命令 有 一 定 的 了 解 。 虽 然 PHP 提供 了 很 多 
的 原生 函数 帮助 对 操作 系统 进行 相应 的 操作 , 但 是 仍旧 有 些 操作 是 PHP 力 有 不 逮 的 。 比 如 ， 
我 们 想 复 制 某 个 指定 的 文件 夹 , 包括 其 中 的 子 文件 夹 和 文件 , 或 者 想 在 PHP 脚本 中 执行 使 
用 其 他 编程 语言 写 出 的 程序 ， 但 PHP 没有 提供 相应 的 原生 函数 ， 这 时 ， 就 只 能 在 PHP 脚 
本 中 嵌入 与 这 些 操作 相关 的 指令 或 程序 。 
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PHP 为 我 们 提供 了 四 种 方法 来 调用 操作 系统 命令 和 执行 使 用 其 他 编程 语言 写 出 的 
程序 。 
口 重音 符 () : PHP 可 以 执行 嵌入 到 一 对 重音 符 〈`) 之 间 的 操作 系统 命令 并 返回 ， 
使 命令 输出 。 这 一 对 重音 符 其 实 是 shell_exec0 函 数 的 简写 形式 。 
口 system0 函 数 : 使 用 PHP 的 system0O 函 数 可 以 执行 一 条 操作 系统 命令 并 返 
最 后 一 行 输出 。 
口 exec0 函 数 : 使 用 PHP 的 exec0 函 数 可 以 执行 一 条 操作 系统 命令 并 将 命令 输出 存 入 
指定 数组 ， 然 后 返回 命令 的 最 后 一 行 输出 。 
口 passthru0 函 数 : 使 用 PHP 的 passthru0 函 数 可 以 执行 一 条 操作 系统 命令 并 返回 命令 

输出 。 

值得 注意 的 是 ， 如 果 需 要 使 用 这 四 项 函数 来 执行 系统 命令 ， 需 要 确认 它们 在 php.ini 
配置 文件 中 没有 被 禁用 。 如 需 开 启 这 四 项 函数 ， 可 以 打开 php.ini， 查 找 “disable_ functions 
=” 关 键 字 ， 然 后 将 等 号 右边 的 shell exec、system、exec 和 passthru 都 去 掉 。 在 保存 修改 
后 的 php.ini 之后， 重启 apache 服务 器 以 便 修 改 生 效 。 


回 


命令 的 


11.2.1 ”重音 符 () 


执行 系统 命令 最 简单 的 一 种 方法 ， 就 是 把 需要 执行 的 命令 放 在 一 对 重音 符 之 间 ， 然 后 
将 其 赋 给 某 个 变量 。 如 果 需 要 输出 该 命令 的 结果 ， 则 可 以 使 用 echo 或 者 print 之 类 的 打印 
命令 字 进 行 输出 。 

例如 ,我 们 想 要 输出 C 盘 下 一 个 名 为 python27 文件 夹 里 的 文件 列表 , 则 可 以 运行 如 下 
于 本 : 


$result = ‘dir c:N\pYython27 `; 

当 PHP 执行 了 这 一 句 脚本 之 后 ， 变 量 $result 中 就 保存 了 系统 命令 “dir c:\python27” 的 
执行 结果 ， 也 就 是 一 串 文 件 夹 和 文件 列表 。 这 时 ,我 们 可 以 使 用 echo 命令 输出 执行 结果 到 
浏览 器 中 ， 得 到 类 似 如 下 的 结果 : 

驱动 器 C 中 的 卷 没 有 标签 。 

卷 的 序列 号 是 E8A0-76E7 


c:\python27 的 目录 


2013=06=-12 19:08 <DIR> 
2013=06=12 19:08 <DIR> Se 
2043=06=12 9208 <DIR> DLLs 


2013=06=12 19208 <DIR> Doc 
20413=06=12 19208 <DIR> include 
2013=06=12 "19:08 <DIR> Lib 
2013506=12 1906 <DIR> libs 
OLS3=05=L9 253 40,098 LICENSE.txt 
2013=05=15, 222>4 让 359,882 NEWS.txt 
2043=05=15 22>543 26, 624 python.exe 
2013=05=15 22:43 27,136 pythonw.exe 
OL3=05=15 22=41 54,979 README .txt 
20.3=06=12 19508 <DIR> 记忆 于 
20.3=06=12° -19=08 <DIR> Tools 
2013=05=15" 22343 49,664 w9xpopen.exe 
6 个 文件 558,383 字 节 
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9 个 目录 25,699,856,384 可 用 字 节 


11.2.2 system() 函 数 、exec() 函 数 和 passthru() 函 数 


除 此 之 外 ， 我 们 还 可 以 使 用 system0 〇 函数 来 调用 系统 命令 ， 输 出 命令 执行 结果 。 与 前 
面 提 到 的 那 一 对 重音 符 不 同 的 是 ， 当 使 用 system0 函 数 调 用 某 系统 命令 时 ， 系 统 就 会 自动 
将 该 命令 的 执行 结果 输出 到 浏览 器 。 如 果 我 们 将 system0 函 数 的 返回 值 赋 给 某 个 变量 ， 并 
将 该 变量 的 值 输 出 到 浏览 器 ， 那 么 将 看 到 命令 执行 结果 的 最 后 一 句 打印 了 两 次 。 因 为 
system( 函 数 返 回 的 值 是 该 命令 执行 结果 的 最 后 一 行 。 

exec() 函 数 与 system() 函 数 的 返回 值 一 样 ， 都 是 在 执行 命令 后 返回 该 命令 执行 结 ee 
后 一 行 。 但 是 exec0 函 数 与 system0 函 数 不 同 的 是 ， pe 命令 执行 结 
到 浏览 器 。 因 此 ， 当 使 用 exec0 函 数 将 某 条 系统 命令 的 执行 绪 po 
量 的 值 输 出 到 浏览 器 ， 我 们 只 会 看 到 该 命令 执行 结果 的 最 后 一 行 。 如 果 需 要 查看 命令 执行 
结果 的 其 他 内 容 ， 可 以 将 通过 er 的 杰 信 的 执行 关 果 保存 到 -个 数组 中 ， 然 后 
通过 foreach 命令 字 输 出 命令 的 执行 结果 


$result = exec("dir c:\python27", $dirout) 


foreach ($dirout as $line) 


echo $line. "<br>"; 
上 面 这 段 脚 本 中 ，$dirout 是 我 们 定义 的 数组 ，exec0 函 数 将 命令 “dir ci\python27” 的 
结果 输出 到 这 个 数组 中 ; 然后 使 用 foreach 遍历 这 个 数组 ,得 到 与 之 前 那 一 对 重音 符 相同 的 

结果 。 当然 ， 我 们 也 可 以 使 用 如 下 的 脚本 输出 指定 行 的 内 容 : 

echo $dirout[3]; 

echo $dirout[7]; 

与 system() 和 execO 函 数 都 不 同 的 是 , passthru0 函 数 不 会 返回 任何 值 , 只 会 自动 输出 命 
令 执 行 结果 到 浏览 器 。 因此， 大 家 不 要 用 passthru0 函 数 来 为 变量 赋值 ， 要 不 然 在 输出 变量 
时 得 到 的 只 能 是 空 值 。 


11.2.3 ”四 个 变量 的 区 别 


下 面 我 们 比较 一 下 这 四 个 变量 ， 如 表 11-2 所 示 。 
表 11-2 shell_exec()、system()、exec() 和 passthru() 对 比 
函数 功能 i 返回 值 内 容 其 他 输出 
shell execO 3 串 ， 命 令 执行 结果 | 无 
， 命 令 执行 结果 的 最 后 | 自动 输出 全 部 执行 结果 至 


Steno 调用 系统 命 | 全] 一行 浏览 器 
令 ， 输 出 命令 ， 命令 执行 结果 的 最 后 | 二 
exec0) 执行 结果 ee 无 
自动 输出 全 部 执行 结果 至 
passthru() 浏览 器 
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我 们 提 到 过 , 在 使 用 这 些 函数 之 前 , 需要 确认 它们 在 php.ini 配置 文件 中 有 没有 被 禁用 。 
为 通过 这 些 函 数 ， 恶 意 用 户 可 能 会 执行 你 不 想 让 他 们 执行 的 命令 或 者 获取 你 不 想 授予 他 
门 的 权限 。 因 此 ， 在 不 了 解 是 否 可 能 会 造成 安全 风险 之 前 ， 建 议 禁用 这 四 个 函数 。 


四 | 


11.3 使 用 PHP 操控 FTP 


在 互联 网 上 ， 每 天 都 会 有 成 千 上 万 的 文件 从 一 台电 脑 传送 到 另 一 台电 脑 。 当 一 家 公司 
不 同 分 部 的 员工 们 想 要 通过 互联 网 传送 文件 ， 那 真是 一 点 儿 问 题 也 没有 。 现 如 今 ， 我 们 可 
以 使 用 的 文件 传输 方式 真是 数不胜数 ， 方 式 虽 多 ， 但 是 不 外 乎 两 种 方式 ， 一 种 是 通过 Web 
服务 器 ， 也 就 是 通过 超 文本 传输 协议 CHITP) ， 另 一 种 则 是 通过 文件 传输 协议 (FTP) 。 
在 之 前 的 章节 里 , 我 们 学 习 了 如 何 通过 PHP 调用 HTTP 上 传 文件 到 服务 器 。 现 在 来 看 看 如 
何 使 用 PHP 调用 FTP 软件 来 传输 文件 。 

FTP 软件 通常 都 是 一 分 为 二 的 客户 端 /服务 器 架构 。 使 用 FTP 软件 在 本 地 电脑 和 远程 电 
脑 之 问 传输 文件 ， 需 要 在 本 地 电脑 安装 FTP 软件 的 客户 端 , 在 远程 电脑 上 安装 FTP 软件 的 
服务 器 端 。 然 后 通过 客户 端 和 服务 器 端 通信 在 本 地 电脑 和 远程 电脑 间 建 立 FTP 连接 ， 从 而 
实现 文件 传输 。 

若 需 要 通过 PHP 脚本 使 用 FTP 协议 , 则 需要 在 安装 了 PHP 引擎 之 后 , 启用 FTP 支持 。 
在 Windows 操作 系统 中 ，FTP 支持 是 默认 开户 的 。 但 在 Linux、UNIX 及 Mac 操作 系统 中 ， 
FTP 支持 需要 手动 开启 。 

学 习 本 章 的 内 容 ， 需 要 架设 FTP 服务 器 。 互 联网 上 开源 并 免费 的 FTP 服务 器 有 很 多 ， 
这 里 我 们 使 用 FileZilla 服务 器 。 需 要 下 载 的 读者 ， 可 以 访问 FileZilla 项 目的 官方 网 站 
Chttps:Wfilezilla-project'org/) 获取 。 


11.3.1 准备 工作 


在 安装 好 FileZilla 之 后 ,双击 桌面 上 的 FileZilla 服务 器 的 图 标 ， 启 动 FileZilla 服务 器 。 
这 时 ， 如 图 11-1 所 示 的 FileZilla 的 管理 界面 也 显示 在 了 桌面 上 。 


图 11-1 FileZilla 管理 登录 界面 
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后 面 


通过 这 个 管理 登录 界面 ， 我 们 可 以 知道 服务 器 的 地 址 (Server Address) 是 127.0.0.1。 
的 端口 号 (port) 是 用 来 对 FTP 服务 器 进行 管理 时 使 用 的 专用 端口 号 ， 不 可 向 外 界 透 


露 。 管 理 密码 〈Administration password) 则 需要 自行 指定 。 在 完成 这 一 切 后 ， 单 击 OK 按 
钮 连接 上 FileZilla 服务 器 。 


图 11-2 ”FileZilla 服务 器 管理 界面 


在 连接 上 FileZilla 服务 器 后 ， 进 入 了 FileZilla 服务 器 的 管理 界面 ， 如 图 11-2 所 示 。 在 
这 里 我 们 可 以 对 服务 器 进行 管理 和 配置 。 

先 在 服务 器 上 建立 一 个 FTP 用 户 ， 并 为 其 分 配 一 个 可 访问 的 目录 。 

(1) 单 击 工具 栏 上 的 “ 轨 ”按钮 ， 弹 出 “用 户 ” 对 话 框 ， 如 图 11-3 所 示 。 


You can enter some comments about the user 


图 11-3 FileZilla 服务 器 用 户 管理 界面 


(2) 在 用 户 管理 界面 右 侧 的 “用 户 (Users) ”区 域 框 里 单 击 Add 按钮 。 在 弹出 的 窗 
口中 输出 用 户 名 johndow， 分 组 信息 为 默认 的 “<none>”。 然 后 ， 单 击 “ 确 认 ” 按 钮 。 
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(3) 在 “用 户 (Users〉 ”区 域 框 右 侧 的 “账户 设置 (Account settings) ”区 域 框 内 勾 
选 密码 (Password), 并 在 其 右 侧 的 文本 框 里 为 上 一 步 中 添加 的 用 户 指定 密码 “Password*”。 

(4) 单 击 用 户 管理 界面 左 侧 的 页 面 区 域 框 里 的 “Shared folders” 页 面 。 然 后 单 击 在 其 
右 侧 出 现 的 “Shared folders” 区 域 框 内 的 “Add” 按 钮 ， 为 用 户 名 johndow 的 用 户 分 配 共 
享 文件 夹 ， 并 为 其 分 配 适 当 的 文件 操作 权限 。 然 后 单 击 窗口 左下 角 的 OK 保存 设置 并 关闭 
用 户 管理 界面 ， 如 图 11-4 所 示 。 


如 分 瑟 文 件 操作 权限 


Doo 添加 用 户 


二 大 | 
> Ei 
ng) [Remove | (Rename ] [Set abome cr] Renome Copy 
A dectoy abss wdl also appear a the speciied location Asses must conian the ful local 
palh Sepsale mbple dhares ot one hectory wah the poe chacacter 人 
I using haces, please avoid cyclic drectony shuctunes, t wil only confuse FTP cherks. 


图 11-4 为 用 户 johndow 设置 共享 文件 夹 并 分 配 文件 操作 权限 
至 此 ， 我 们 的 准备 工作 就 做 好 了 。 


11.3.2 登录 FTP 服务 器 


在 登录 FileZilla 服务 器 之 前 , 需要 在 本 地 电脑 和 FTP 服务 器 间 建 立 一 条 FTP 连接 。 在 
PHP 脚本 中 ， 可 以 使 用 ftp_connect0 函 数 来 实现 : 

$connect = ftp connect('127.0.0.1'); 

如 果 我 们 为 FileZilla 服务 器 配置 了 域名 的 话 , ftp_connect0 的 参数 也 可 以 是 FileZilla 服 
务 器 的 域名 。 如 : 

$connect = ftp connect ('www.example.com'); 

执行 该 脚本 ， 系 统 没 有 提示 任何 错误 ， 则 表明 FTP 连接 成 功 建立 了 。 

接 下 来 ， 我 们 需要 向 FTP 服务 器 提交 用 户 名 和 密码 用 于 验证 身份 。 这 时 就 需要 使 用 
ftp_login0 函 数 ， 该 函数 有 三 个 必要 参数 ， 分 别 是 FTP 连接 对 象 、 用 户 名 和 密码 。 

$login _ result = ftp login($connect, 'johndow','PasswOrd*'); 

如 果 我 们 在 FTP 连接 没有 建立 的 情况 下 执行 fp login0 函 数 ， 系 统 会 发 出 警告 ， 同 时 
继续 执行 余下 的 脚本 。 为 了 防止 余下 的 脚本 不 会 在 FTP 连接 没有 建立 的 情况 下 执行 我们 
可 以 把 建立 连接 的 脚本 修改 为 : 
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$connect = ftp connect('127.0.0.1') or die(" 找 不 到 服务 器 。 ' ) 
或 者 
$connect = ftp connect ('www.example.com') or die (' 找 不 到 服务 器 。"' ) ; 


同样 地 ， 为 了 防止 余下 的 脚本 不 会 在 没有 成 功 登 录 的 情况 下 执行 ， 我 们 也 可 以 在 登录 
服务 器 的 脚本 修改 为 : 


$login _ result = ftp login($connect, 'johndow', 'Passw0rd*') or die(' 无 法 
登录 服务 器 。' ) ; 


当 我 们 执行 登录 脚本 后 ， 若 系统 没有 发 生 任何 告警 或 错误 提示 ， 表 明成 功 登 录 了 
FileZilla 服务 器 。 现 在 我 们 就 可 以 获取 服务 器 上 的 文件 列表 、 向 服务 器 上 传 文件 或 从 服务 
器 下 载 文件 了 。 


11.3.3 ”获取 服务 器 文件 列表 


在 FTP 服务 器 上 最 常见 的 操作 是 获取 文件 列表 。 为 此 ，PHP 为 我 们 准备 了 ftp_nlistO 
函数 来 实现 这 项 操作 。 该 函数 有 两 个 必要 参数 ， 分 别 为 FTP 连接 对 象 和 需 获取 文件 列表 对 
应 的 文件 夹 名 。 在 这 里 ， 我 们 需要 列 出 分 配给 “johndow” 用 户 的 “johndow _filezilla_ ftp” 
文件 夹 中 的 文件 列表 ， 脚 本 如 下 : 


$filesArr = ftp nlist($connect, "johndow_ filezilla ftp'); 


这 时 ， 指 定 文件 夹 中 的 文件 列表 就 以 数组 的 形式 存 入 到 了 S$filesArr 中 。 

如 果 我 们 并 不 知道 分 配给 johndow 用 户 的 文件 夹 叫 什么 名 字 ， 则 可 以 使 用 ftp_pwd0) 
函数 来 获取 当前 文件 夹 。 当 以 johndow 用 户 的 身份 登录 服务 器 时 ， 当 前 文件 夹 就 是 之 前 我 
们 分 配 johndow 用 户 的 文件 夹 。 因 此 ， 上 一 句 脚本 可 以 修改 为 : 

$directory = ftp pwd($connect); 

$filesArr = ftp nlist($connect, $directory); 

之 后 ， 我 们 就 可 以 像 遍历 普通 数组 的 方式 输出 当前 文件 夹 下 的 文件 列表 : 


foreach ($filesArr as S$value) 
人 
echo $value.'<br>'; 


11.3.4 下 载 和 上 传 文件 


我 们 可 以 使 用 ftp_getO 函 数 从 FileZilla 服务 器 下 载 文件 。 该 函数 有 四 个 参数 ， 分 别 是 
FTP 连接 对 象 、 文 件 下 载 到 本 地 后 的 文件 名 、 文 件 在 FileZilla 服务 器 上 的 文件 名 以 及 文件 
类 型 。 在 这 里 我 们 需要 将 FileZilla 服务 器 上 的 fle01.txt 文件 下 载 到 本 地 ， 并 以 newfile.txt 
的 文件 名 进行 保存 ， 则 脚本 可 以 写成 : 


ftprget (sconmmnect7 "newtile txt 7 File0l tt FTPIASCLIINS 
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对 于 ftp_getO 函 数 的 最 后 一 个 参数 的 “文件 类 型 ”只 有 两 种 ， 一 种 为 FTP ASCII， 另 
一 种 为 FTP BINARY。 前 者 指 的 是 如 文本 文件 之 类 的 可 以 不 需 借助 其 他 阅读 工具 就 可 以 阅 
读 的 文件 , 而 后 者 则 指 的 是 余下 的 其 他 类 型 的 文件 。 这 里 我 们 需要 下 载 的 是 锯 e01.txt 文件 ， 
因此 ， 文 件 类 型 应 该 指定 为 FTP_ASCII。 

在 运行 了 上 面 的 脚本 之 后 ， 若 fe01.txt 文件 确实 存在 ， 则 会 被 下 载 到 本 地 目录 中 ， 并 
被 重 命名 为 newfile.txt。 

与 从 服务 器 下 载 文件 类 似 ， 向 服务 器 上 传 文件 需要 用 到 ftp_put0 函 数 。 该 函数 同样 也 
有 四 个 参数 ， 分 别 是 FTP 连接 对 象 、 文 件 上 传 后 在 服务 器 上 的 文件 名 、 文 件 在 本 地 的 文件 
名 及 文件 类 型 。 在 这 里 我 们 需要 将 本 地 的 newfile.txt 上 传 到 FileZilla 服务 器 上 ， 则 脚本 可 
以 写成 : 

ftp put ($connect, "newfile.txt"， 'newfile.txt', FTP ASCII); 

若 本 地 确实 存在 一 个 名 为 newfile.txt 的 文件 ， 则 在 脚本 执行 后 ， 该 文件 会 被 上 传 到 
FileZilla 服务 器 上 并 以 newfile.txt 的 名 称 进行 存储 。 


11.3.5 ”使 用 PHP 操控 FTP 


需要 注意 的 是 ， 在 完成 了 对 FileZilla 服务 器 的 操作 之 后 ， 需 要 使 用 ftp_close0 函 数 断 
开 FTP 连接 。 该 函数 只 有 一 个 参数 ， 那 就 是 FTP 连接 对 象 。 

现在 我 们 把 本 节 学 习 的 内 容 总 结 一 下 。 

【 例 11.3】 使 用 PHP 操控 FTP。 


下 <?php 
也 $connect = ftp connect ("127.0.0.1") // 连 接 服务 器 
3 or die(" 找 不 到 服务 器 ") ; 
4 $login result = ftp login($connect, 'johndow', 'PasswOrd*') 
// 登 录 服务 器 
5 or die ("无 法 登录 服务 器 ") ; 
6 
7 $directory = ftp pwd($connect); // 获 取 当 前 目录 
8 $filesArrBefore = ftp nlist($connect, $directory); 
// 获 取 当 前 目录 的 文件 列表 
9 
10 fileList ($filesArrBefore); // 输 出 当前 目录 的 文件 列表 
2 ftp get ($connect, 'newfile.txt', "file01.txt'，FTP ASCII); 
// 下 载 file01 .txt 到 本 地 
3 
14 ftp put ($connect, 'newfile.txt', 'newfile.txt', FTP ASCII); 
// 上 传 newfile .txt 文件 到 服务 器 
15 
16 echo "<hr>'"7 
17 
18 $filesArrAfter = ftp nlist($connect, $directory); 
// 青 次 获取 当前 目录 的 文件 列表 
3 
20 fileList ($filesArrAfter); // 输 出 当前 目录 的 文件 列表 
这 出 
22 ftp close($connect); // 关 闭 当 前 FTP 连接 
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23 

24 function fileList(S$filesRrr) // 遍 历 函 数 

25 i 

26 foreach ($filesArr as S$value) 

Eh | 

28 echo $value."<br>"; 

29 

30 } 

号 1 2> 

在 运行 了 例 11.3 中 的 脚本 之 后 ， 可 以 发 现 浏览 器 中 列 出 了 两 份 文件 列表 ， 其 中 后 一 份 
列表 比 前 一 份 多 出 了 一 个 文件 newfiletxt。 而 本 地 和 服务 器 的 目录 中 则 分 别 多 出 了 两 个 名 
为 newfile.txt 的 文件 。 


11.4 使 用 PHP 发 送 电 子 邮件 


当今 电子 邮件 是 使 用 最 为 广泛 的 互联 网 应 用 之 一 。 许 多 的 PHP 应 用 程序 都 会 要 求 发 送 
电子 邮件 到 用 户 邮 件 来 验证 用 户 身份 ;或 者 向 用 户 发 送 订单 确认 信息 等 。 例 如 ， 在 许多 的 
网 站 上 会 有 “忘记 密码 ”这 个 链接 ， 单 击 这 个 链接 ， 输 入 相应 的 信息 。 不 久 后 ， 就 会 收 到 
一 封 电子 邮件 帮助 找 回 丢失 的 密码 。 电 子 邮件 的 应 用 可 谓 层 出 不 穷 。 那 么 我 们 如 何 才能 使 
用 PHP 来 发 送 电子 邮件 和 附件 呢 ? 

PHP 提供 了 一 些 函 数 ， 可 以 让 我 们 简便 地 发 送 电子 邮件 。 在 本 节 里 ， 我 们 将 了 解 如 何 
使 用 PHP 脚本 来 发 送 电子 邮件 和 附件 。 


11.4.1 准备 工作 


在 使 用 PHP 发 送 电子 邮件 之 前 ， 我 们 需要 搭建 邮件 服务 器 。 与 FTP 服务 器 类 似 ， 网 
络 上 也 有 许多 开源 的 邮件 服务 器 。 在 这 里 我 们 使 用 国产 的 Magic Winmail 服务 器 来 演示 如 
何在 本 地 搭建 和 配置 邮件 服务 器 。 更 多 的 内 容 可 以 参见 Magic Winmail 的 官方 网 站 
(http://www.magicwinmail.com/docs/) 支撑 文档 。 

在 下 载 并 安装 好 了 Magic Winmail 服务 器 后 ， 启 用 Winmail 服务 器 程序 。 当 系统 托盘 
处 出 现 国 图 标 时 ， 表 示 邮 件 服务 器 已 经 启用 。 首 次 启用 邮件 服务 器 时 会 出 现 设置 向 导 ， 
关闭 即 可 。 双 击 桌 面 上 的 Magic Winmail 管理 端 工具 图 标 ， 弹 出 如 图 11-5 所 示 的 “连接 服 
务 器 ”窗口 。 


11-5 ”Magic Winmail 管理 端 工具 登录 窗口 
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(1) 在 “登录 用 户 ” 区 域 框 的 “密码 ”处 输入 在 安装 过 程 中 设置 的 管理 密码 ， 然 后 单 
击 “ 确 定 ” 按 钮 登录 Winmail 邮件 服务 器 。 成 功 登 录 服务 器 后 弹出 图 11-6 所 示 的 对 话 框 。 


图 11-6 成 功 登 录 Magic Winmail 服务 器 管理 端 工具 


(2) 展开 左 树 的 “域名 设置 ”， 单 击 “ 域 名 管理 ” 子 项 为 邮件 服务 器 添加 主 域 。 在 单 
击 了 “域名 管理 ”页 面 左下 角 的 “新 增 ” 按 钮 之 后 ， 弹 出 “域名 ”窗口 。 这 时 ， 在 “基本 
参数 ”页 签 下 的 “域名 ”文本 框 里 输入 “localhost.com”， 然 后 单 击 “ 确 定 ” 按 钮 。 至 此 ， 
服务 器 的 主 域 添 加 完成 ， 如 图 11-7 所 示 。 


默认 流量 控制 | 默认 密码 策略 | 邮箱 限制 ‖ 签名 档 | 
| 高 级 属 性 。 | 默认 容量 | 默认 权限 | 


所 eolhost onl 


邮箱 密码 加 密 方 式 |MD5 

口 本 域 下 增加 新 用 户 自动 增加 到 everyone 组 
口 本 域 下 的 用 户 增加 个 人 信息 到 公用 地 址 湾 
口 用 户 可 以 决定 个 人 信息 是 否 到 公用 地 址 簿 
口 公用 地 址 浒 只 允 放 本 域 下 的 用 户 访问 


图 11-7 为 邮件 服务 器 添加 主 域 


(3) 展开 左 树 的 “用 户 和 组 ”， 单 击 “ 用 户 管理 ” 子 项 添加 用 户 。 系 统 中 已 经 默认 存 
在 一 个 管理 员 用 户 postmaster。 单 击 “ 用 户 管理 ”页 面 左下 角 的 “新 增 ” 按 钮 ， 在 弹出 的 
“基本 设置 ”对 话 框 内 输入 用 户 信息 ， 然 后 单 击 “ 完 成 ”按钮 。 这 时 “用 户 管理 ”页 面 上 列 
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出 了 刚才 添加 的 用 户 ， 如 图 11-8 所 示 。 


图 11-8 添加 用 户 


(4) 然后 打开 PHP 引擎 的 php.ini 文件 ， 找 到 “[mail function]”。 将 其 项 下 的 “SMTP” 
设置 为 “localhost”，“sendmail from” 设 置 为 pm com”。 然 后 检查 
一 下 这 两 项 参数 前 面 是否 有 分 号 (;) ， 若 有 ， 则 删除 这 些 分 号 ， 如 图 11-9 所 示 。 随 后 ， 
重启 Apache 服务 器 。 


intt。 

You set this walue to a high nunber yy a 
tack and eventually crash PHP (out to reaching the 
pq by the Operating Systen). 
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图 11-9 修改 php.ini 


(5) 现在 打开 浏览 器 ， 在 地 址 框 内 输入 “http://127.0.0.1:6080”， 弹 出 Magic Winmail 
的 Webmail 界面 。 然 后 输出 之 前 添加 的 用 户 John Dow 的 用 户 名 和 密码 ， 单 击 “ 登 录 >>>” 
按钮 。 

在 Webmail 的 首页 我们 可 以 看 到 John Dow 用 户 的 邮箱 内 有 一 封 未 读 邮 件 。 记 住 这 
个 数字 , 因为 待 会 儿 将 使 用 PHP 脚 本 向 John Dow 的 邮件 再 发 送 一 封 邮 件 。 若 到 时 ,John Dow 
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的 邮箱 内 有 两 封 未 读 邮 件 ， 则 表示 我 们 通过 PHP 脚本 向 John Dow 发 送 邮件 成 功 了 。 
11.4.2 ”发 送 电子 邮件 


在 PHP 脚本 中 ， 我 们 可 以 使 用 mail0 函 数 向 指定 用 户 发 送 邮 件 。 这 个 函数 有 三 个 必 选 
参数 和 一 个 可 选 参数 。 格 式 如 下 : 


mail (address, subject, message, headers); 


这 四 个 参数 的 说 明 如 下 : 
口 address 指 的 是 收 件 人 的 邮箱 地 址 。 在 这 里 就 是 John Dow 的 邮箱 地 址 。 
口 subject 指 的 是 邮件 的 主题 。 
口 message 指 的 是 邮件 的 内 容 。 
口 headers 指 的 是 邮件 头 信息 。 
在 这 四 个 参数 中 ， 前 三 个 参数 是 必 选 的 ， 第 四 个 参数 是 可 选 的 。 
现在 我 们 就 来 看 看 如 何 使 用 PHP 脚本 向 John Dow 的 电子 邮箱 发 送 一 封 邮件 。 
【 例 11.4】 使 用 mailO 函 数 向 指定 用 户 的 电子 邮箱 发 送 电子 邮件 。 
<?php 
$to = "johndow@localhost .com"; 
$subj = "test"; 
Smess = "This is a test of the mail function"; 


S$headers = "bcc:techsupport@localhost.com\r\n"; 
Smailsend = mail ($to,$subj, $mess, $headers); 


ww 


2> 


在 上 面 的 这 段 脚 本 中 设置 了 四 个 变量 ， 分 别 为 Sto、$subj、$mess 和 $headers。 这 四 个 
参数 分 别 对 应 收 件 人 地 址 、 邮 件 主题 、 邮 件 内 容 和 邮件 头 信息 。 最 后 一 句 使 用 了 mail0 函 
数 实现 了 邮件 的 发 送 。 

运行 了 上 面 这 段 脚 本 后 ， 我 们 再 登录 John Dow 的 Webmail 邮箱 。 查 看 一 下 John Dow 
的 收 件 箱 里 是 不 是 收 了 一 封 来 自 postmaster@localhost.com 的 邮件 呢 ? 如 图 11-10 所 示 。 


Winmail 


图 11-10 ”检查 用 户 的 收 件 箱 ， 确 认 发 送 结果 
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我 们 惊喜 地 发 现 ，John Dow 的 收 件 箱 里 的 确 多 了 一 条 来 自 postmaster@localhost.com 
的 主题 为 “test” 邮 件 。 说 明 ， 我 们 使 用 PHP 脚本 发 送 电子 邮件 成 功 了 。 


11.4.3 发送 带 附 件 的 电子 邮件 


现 如 今 , 大 家 都 喜欢 在 邮件 里 发 个 附件 。 那 么 使 用 PHP 脚本 发 送 带 附件 的 电子 邮件 要 
如 何 做 呢 ? 这 里 假设 要 在 邮件 里 带 上 一 张 动画 图 片 ， 具 体 如 何 实现 请 看 下 面 的 解析 。 
首先 ， 我 们 先 来 看 一 份 电子 邮件 在 传送 过 程 中 的 样子 : 


Return-Path: 

Date: Mon, 22 May 2000 19:17:29 +0000 

From: Someone 

To: Person 

Message-id: <83729KI93LI9214@example.com> 

Content-type: multipart/mixed; boundary="396d983d6b89a" 

Subject: Here's the subject 

--396d983d6b89a // 邮 件 头 信息 到 此 结束 
Content-type: text/plain; charset=iso-8859-1 
Content-transfer-encoding: 8bit 


This is the body of the email. 


--396d983d6b89a // 邮 件 内 容 到 此 结束 
Content-type: text/html; name=attachment .html 
Content-disposition: inline; filename=attachment .html 
Content-transfer-encoding: 8bit 


This is the attached HTML file 


--396d983d6b89a 一 // 附 件 内 容 到 此 结束 


在 这 封 邮件 中 ， 前 七 行为 邮件 头 信 息 。 在 邮件 头 信息 中 ， 我 们 发 现 Content-type 为 
multipart/mixed， 而 boundary 为 一 随机 字符 串 。 这 一 随机 字符 串 做 为 分 隔 邮 件 头 和 邮件 主 
体 、 以 及 邮件 主体 中 的 邮件 内 容 和 附件 内 容 的 分 隔 符 。 注 意 该 随机 字符 串 出 现 了 三 次 ， 其 
中 ， 第 一 次 和 第 二 次 出 现时 ， 其 前 面 加 上 了 “--” 符 号 ， 而 最 后 一 次 出 现时 ， 其 前 后 都 加 
wi 

如 果 需 要 发 送 带 附件 的 邮件 ， 则 需要 分 别 构造 邮件 头 信息 和 邮件 主体 。 

【 例 11.$】 使 用 mailO 函 数 向 指定 用 户 发 送 带 附件 的 电子 邮件 。 


<?php 
// 定 义 收 信人 
$to = 'johndow@localhost .com'; 
// 定 义 邮件 主题 
$subject = 'Test email with an image attached'; #1 
// 创 建 邮 件 部 件 边界 
// 在 这 里 ， 我 们 使 用 md5 () 函数 ， 以 当前 时 间 为 对 象 生 成 一 个 唯一 随机 数 
Srandom hash = md5(date('r', time())); #2 
// 定 义 需要 传送 的 邮件 头 信息 。 注 意 邮件 头 之 间 用 “\rz\n” 隔 开 
0 S$headers = "From: podmaster@localhost.com\r\nReply-To: podmaster@ 
localhost .com"; 


11 // 添 加 邮件 部 件 边界 和 MIME 类 型 
12 $headers .="\r\nContent-Type: multipart/mixed; boundary=\"PHP-mixed-"- 


户 
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Srandom hash.™\""; #3 
13 // 读 取 附 件 文件 内 容 到 一 个 字符 串 变 量 
14 // 用 base64_encode() 函数 对 该 字符 串 变 量 进行 编码 


15 “// 然 后 将 编码 后 的 内 容 分 割 成 若干 小 块 以 利 传输 
16 S$attachment = chunk split (base64 encode (file get contents('mouth.gif"))); 


#4 
17 // 定 义 邮件 内 容 
18 ob_start (); // 打出 输出 至 缓存 选项 #5 
0 2> 
20 


21 --PHP-mixed-<?php echo $random hash; ?> 
22 Content-Type: multipart/alternative; boundary="PHP-alt-<?php echo 


$random hash; ?>" #6 
23 
24 --PHP-alt-<?php echo $random hash; ?> 
25 Content-Type: text/plain; charset="iso-8859-1" #7 
26 Content-Transfer-Encoding: 7bit 
27 


28 Hello World!!! 
29 This is simple text email message. 


30 

31 --PHP-alt-<?php echo S$random hash; ?> 

32 Content-Type: text/html; charset="iso-8859-1" #8 
33 Content-Transfer-Encoding: 7bit 

34 


35 <h2>Hello World!'</h2> 
36 <p>This is something with <b>HTML</b> formatting.</p> 


37 
38 --PHP-alt-<?php echo $random hash; ?>-- #9 
39 
40 --PHP-mixed-<?php echo Srandom hash; ?> #10 


41 Content-Type: image/gif; name="mouth.gif" 
42 Content-Transfer-Encoding: base64 
43 Content-Disposition: attachment 


44 

45 <?php echo $attachment; ?> 

46 --PHP-mixed-<?php echo S$random hash; ?>-- #11 
47 

48 <?php 

49 // 将 当前 缓存 内 容 存 入 到 变量 $message 中 ， 然 后 清空 缓存 

50 Smessage = ob get clean(); #12 
51 // 发 送 电 子 邮 件 

52 S$mail sent = Qmail( $to, $subject, $message, S$headers ); #13 
53 ”// 车 邮件 发 送 成 功 ， 显 示 “Mail sent”。 反 之 则 显示 “Mail failed.” 

54 echo $mail sent ? "Mail sent" : "Mail failed"; 

EL 


运行 上 面 这 段 脚本 之 后 ， 我 们 再 次 登录 John Dow 的 Webmail 邮箱 ， 发 现 其 收 件 箱 中 
多 出 了 一 封 名 为 “Test email with an image attached” 的 邮件 。 打 开 该 邮件 ， 赫 然 发 现 邮 件 
正文 和 附近 都 发 送 成 功 了 。 而 且 附 件 还 以 签名 的 形式 显示 在 正文 的 下 方 ,如 图 11-11 所 示 。 

上 面 这 段 脚本 一 共 由 两 对 “<?php .. ?>” 标 记 对 分 隔 开 的 三 部 分 组 成 。 在 第 一 对 
“<?php .… ?>” 标 记 对 中 ， 定 义 了 收 件 人 、 邮 件 主题 和 邮件 头 ， 创 建 了 邮件 部 件 边 界 ， 读 
取 附 件 文件 内 容 并 对 其 进行 编码 ;在 第 二 对 “<?php .… ?>” 标 记 对 中 ， 定 义 了 邮件 正文 的 
内 容 并 发 送 了 邮件 ,在 这 两 对 “<?php .… ?>” 标 记 对 之 间 ， 我 们 撰写 了 邮件 的 正文 和 附件 。 
按照 顺序 把 这 三 部 分 的 内 容 分 别 叫 做 : 
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口 邮件 起 始 部 分 ; 
口 邮件 正文 与 附件 ; 

口 邮件 发 送 部 分 。 

需要 注意 的 是 ， 在 脚本 中 使 用 ob_start0 开 启 了 “输出 至 缓存 ”选项 。 因 此 ， 所 有 的 内 
容 都 会 被 直接 发 送 到 缓存 中 存储 ， 不 再 经 过 脚本 处 理 。 如 果 ob_start0 之 后 的 各 行内 容 前 有 
空格 或 者 不 相干 的 字符 ， 这 些 空格 和 不 相干 的 字符 也 会 被 送 进 缓存 ， 进 而 破坏 邮件 内 容 ， 
导致 在 邮件 中 看 不 到 相应 的 内 容 。 另 外 ， 在 需要 缓存 的 内 容 处 理 完 成 后 ， 务 必 使 用 
ob_get_clean(0) 函 数 获 取 缓存 的 数据 并 清空 缓存 。 
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图 11-11 收 到 通过 PHP 发 送 的 带 附 件 的 电子 邮件 


现在 我 们 就 来 一 步 一 步 地 了 解 如 何 构造 一 封 带 附 件 的 电子 邮件 。 

第 一 步 ”构建 邮件 的 起 始 部 分 

在 这 一 部 分 里 ， 我 们 重点 关注 例 11.5 所 示 脚 本 中 的 标记 为 “ 志 ” 一 “的 ”的 内 容 。 

口 标记 为 “#1” 的 行 与 其 上 一 行 定义 了 收 件 人 和 邮件 主题 。 

口 标记 为 “# 李 ”的 行 定义 了 一 个 随机 数 ， 以 供 构造 邮件 部 件 边 界 。 

口 标记 为 “ 妆 ” 的 行 及 其 上 一 行 定 义 了 邮件 头 。 

注意 , 邮件 头 信 息 有 许多 种 。 在 这 两 行 里 ,我们 看 的 邮件 头 字段 有 “From”、“Reply-To”、 
“Content-Type” 和 “boundary”。 前 三 个 字段 的 字段 名 与 字段 值 之 间 用 的 是 冒号 (:) ， 而 
“boundary” 字 段 的 字段 名 与 字段 值 之 间 用 的 则 是 等 号 〈=) 。 由 于 在 这 封 邮件 中 既 有 正文 
也 有 附件 ， 所 以 这 里 的 Content-Type 应 该 定义 为 “multipart/mixed”， 表 示 邮 件 是 多 个 部 
件 混合 而 成 。 在 这 段 脚本 中 ， 字 段 名 与 字段 值 之 间 使 用 等 号 连接 的 邮件 头 字段 有 
“boundary”、“charset” 和 “name”， 分 别 代表 邮件 部 件 边界 、 字 符 集 和 附件 文件 名 。 

口 标记 为 “ 覆 ” 的 行 读 取 了 需要 传送 的 附件 ， 并 对 其 进行 了 编码 和 分 块 处 理 。 

这 里 使 用 到 了 三 个 函数 ， 分 别 是 file_get_contents()、base64_encode() 和 chunk split()。 
函数 file_get_contents0) 将 附件 的 内 容 读 取 到 一 个 字符 串 中 ， 然 后 交 由 base64_encode() 函 数 
进行 编码 ， 编 码 后 的 内 容 再 交 由 chunk_split0 函 数 进行 分 块 ， 最 后 得 到 $attachment 字符 串 。 
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口 标记 为 “#” 的 行 开 启 了 PHP 引擎 的 “输出 至 缓存 ”选项 。 

开启 该 选项 后 ， 其 后 的 内 容 将 会 被 直接 送 至 缓存 。 

第 二 步 ”构建 邮件 的 正文 和 附件 

在 这 一 部 分 里 ， 我 们 需要 构建 的 正文 和 邮件 的 附件 都 会 以 字符 串 的 形式 送 进 缓存 。 

口 标记 为 “#6” 的 行 往 下 至 第 38 行为 邮件 的 正文 。 

在 这 一 部 分 里 , 我 们 构建 了 两 个 可 选 的 子 部 分 。 若 邮件 服务 器 支持 HTML 代码 的 显示 ， 
则 以 HTML 代码 显示 正文 ， 和 否则 以 纯 文 本 的 形式 显示 正文 。 因 此 ， 在 标记 为 “大 ”的 行 中 
我 们 定义 的 “Content-Type” 为 “multipartalternative” 表 示 往 下 的 内 容 为 可 选 ， 同 时 也 定 
义 了 一 个 新 的 边界 ， 用 来 分 隔 可 选 内 容 。 

口 标记 为 “#7” 和 “#8” 的 行 分 别 定义 了 两 个 可 选 的 部 分 。 

前 者 为 以 纯 文本 方式 显示 正文 的 内 容 ， 所 以 其 “Content-Type” 为 “text/plain”， 而 后 
者 则 以 HTML 代码 的 方式 显示 正文 的 内 容 ， 所 以 其 “Content-Type” 为 “text/html”。 在 这 
两 个 可 选 的 部 分 里 ， 我 们 使 用 的 字符 集 都 是 iso-8859-1， 向 下 兼容 ASCII。 另 外 ， 在 这 两 个 
部 分 里 ， 我 们 还 使 用 了 一 个 新 的 字段 “Content-Transfer-Encoding” 用 来 定义 chunk split() 
函数 分 块 的 大 小 。 

口 标记 为 “#9” 的 行 定义 了 一 个 可 选 部 分 边界 ， 并 以 “--” 结 尾 。 标 志 着 青 无 其 他 可 

选 部 件 。 

口 标记 为 “#10” 的 行 定义 了 一 个 邮件 部 件 边界 ， 开 始 构 建 另 一 个 邮件 部 件 。 

从 该 行 始 往 下 至 第 45 行为 邮件 附件 的 内 容 。 由 于 附件 文件 为 一 张 GIF 图 片 ， 因 此 
“Content-Type” 为 “image/gif”， 同 时 还 定义 了 一 个 新 的 字段 “name”， 供 邮件 服务 器 获 
取 附 件 文件 的 名 称 。 由 于 之 前 我 们 使 用 base64_encode0 函 数 对 读 取 的 附件 文件 内 容 进行 编 
码 ， 因 此 ，“Content-Transfer-Encoding” 为 “base64”。 在 这 一 部 分 里 ， 我 们 还 定义 了 一 
个 新 的 字段 “Content-Disposition”， 该 字段 的 值 告诉 邮件 服务 器 如 何 处 理 当 前 部 件 的 内 容 。 
这 里 我 们 将 “Content-Disposition” 定 义 为 “attachment”， 就 是 告诉 邮件 服务 器 将 当前 部 
分 做 为 邮件 附件 进行 处 理 。 

口 标记 为 “的 1” 的 行 定义 了 一 个 邮件 部 分 边界 ， 并 以 “--” 结 尾 ， 标 志 着 再 无 其 他 

邮件 部 件 。 邮 件 内 容 〈 包 括 正文 和 附件 ) 到 此 结束 。 

至 此 ， 邮 件 正文 和 附件 构建 完毕 。 需 要 再 次 提醒 注意 的 是 ， 写 入 缓存 的 各 行内 容 前 不 
得 有 空格 和 其 他 的 字符 。 如 果 有 的 话 ， 这 些 空格 和 字符 也 会 被 送 入 缓存 ， 破 坏 邮 件 内 容 ， 
造成 邮件 正文 和 附件 无 法 正常 显示 。 

第 三 步 ” 构 建 邮件 发 送 部 分 

在 这 一 部 分 里 ， 我 们 需要 从 缓存 中 获取 邮件 的 内 容 (包括 正文 和 附件 ) ， 然 后 使 用 
mail0 函 数 发 送 邮 件 。 

口 标记 为 “#12” 的 行使 用 了 ob_get_clean0 函 数 从 缓存 中 获取 数据 并 清空 缓存 。 从 组 

存 中 获取 的 数据 被 存放 在 $message 变量 中 。 

口 标记 为 “#13” 的 行使 用 了 mailO 函 数 发 送 了 邮件 。 若 邮件 发 送 成 功 ， 则 显示 “Mail 

sent”; 若 失败 ， 则 显示 “Mail failed”。 
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正如 我 们 在 之 前 的 章节 中 看 到 的 那样 ，PHP 允许 用 两 种 不 同 的 风格 编写 脚本 : 一 种 是 
基于 过 程 的 脚本 编写 风格 (procedure programing) ， 而 另 一 种 则 是 基于 对 象 的 脚本 编写 风 
格 (object-oriented programming) 。 读 者 可 以 选择 其 中 的 一 种 风格 开始 编写 脚本 。 当 然 ， 


如 果 愿 意 的 话 ， 


也 可 以 在 基于 对 象 的 脚本 中 引入 基于 过 程 的 脚本 或 者 在 基于 过 程 的 脚本 中 


编 入 基于 对 象 的 脚本 。 

在 本 书 的 前 面 儿 章 中 ， 大 多 数 的 例 程 都 是 基于 过 程 的 脚本 。 但 是 ， 在 第 8 章 、 第 9 章 
和 第 10 章 的 例 程 和 实战 练习 里 , 我 们 使 用 的 又 是 基于 对 象 的 脚本 编写 风格 。 如 果 读 者 还 是 
不 清楚 两 者 的 区 别 ， 我 们 来 看 看 下 面 两 段 脚本 。 


【 例 12.1】 基于 过 程 的 编程 风格 与 基于 对 象 的 编程 风格 。 
<?php 
// 基 于 过 程 的 编程 风格 
echo '<h3> 基 于 过 程 的 编程 风格 </h3>"'; 
$Sa= 5; 
$b= 6; 
echo '<h4>a = '.$a.', b= '.$b.'</h4>'; 
echo 'a+b = '.($a + $b).'<br>'; 
echo 'a- b= '.($a - $b).'<br>'; 
echo 'axb = '.($a * $b).'<br>'; 
echo 'a /b= '.($a / $b).'<br>'; 
$a = 10; 
$b = 15; 
echo '<h4>a = '.$a.', b= '.$b.'</h4>'; 
echo 'a+b = '.($a + $b).'<br>'; 
echo 'a-=- b= '.($a - $b).'<br>'; 
echo 'a xb = "'.($a* $b).'<br>'; 
echo va b= (da by <bE>7 


在 上 面 这 段 脚 本 中 ,可 以 看 到 我 们 先 定义 了 两 个 变量 , 然后 使 用 echo 命令 一 行 一 行 地 
输出 了 它们 自身 的 值 以 及 这 两 个 变量 的 和 差 积 商 ;接着 我 们 又 重新 定义 了 这 两 个 变量 的 值 ， 
然后 再 一 次 地 输出 了 这 两 个 变量 在 重新 定义 后 的 值 及 它们 的 和 差 积 商 。 看 看 脚本 中 加 粗 的 
部 分 ， 发 现 所 有 的 计算 都 是 在 行 间 即 时 完成 的 。 

现在 再 来 看 一 段 使 用 基于 对 象 的 编程 风格 实现 相同 功能 的 脚本 : 


include once('class/class.calc.php'); 


// 基 于 对 象 的 编程 风格 


$calc = new calc(); 


echo 


'<h3> 基 于 过 程 的 编程 风格 </h3>"; 
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echo "<h4>a 


'.$calc->getVarA().', b= '.$calc->getVarB().'</h4>'; 


echo 'a+ b= '.$calc->add().'<br>"'; 

echo 'a - b = '.$calc->minus().'<br>'; 
echo 'a x b = '.$calc->multiply().'<br>"'; 
echo 'a / b = '.$calc->divide().'<br>'; 


// 使 用 calc 实例 的 setVarA 和 setVarB 方法 来 设置 私有 变量 的 值 
$calc->setVarA (10); 
$calc->setVarB (15); 


echo '<h4>a 
echo "a + b 


'.$calc->getVarA().', b= '.$calc->getVarB().'</h4>'; 
'.$calc->add () .'<br>'; 
1.$calc->minus () .'<br>'7 


echo "a - b 
echo 'axb .$calc->multiply () .'<br>"'; 
echo "a / b -.$calc->divide() .'<br>'7 


Ss 


在 比较 两 段 脚本 之 后 ， 可 以 发 现 第 二 段 脚 本 除了 在 加 粗 的 部 分 上 与 第 一 段 脚本 不 同 之 
外 ， 其 他 都 是 一 样 的 ， 并 且 第 二 段 脚 本 中 加 粗 的 部 分 比 第 一 段 脚 本 中 加 粗 的 部 分 看 上 去 要 
复杂 的 多 。 首 先 ， 我 们 定义 了 一 个 calc 类 的 实例 ， 并 将 其 命名 为 gcalc; 然后 使 用 $calc 实 
例 的 getVarA 和 getVarB 方法 获取 了 calc 类 的 内 置 变 量 $a 和 $b 的 值 ， 接 着 使 用 $calc 实例 
的 add、minus、mnuliply 和 divide 方法 输出 这 两 个 内 置 变量 的 和 差 积 商 。 接 着 ， 使 用 $calc 
实例 的 setVarA 和 setVarB 方法 重新 设置 了 变量 $a 和 Sb 的 值 ， 然 后 再 次 使 用 add、minus、 
mutiply 和 divide 方法 输出 这 两 个 内 置 变 量 的 和 差 积 商 ,所 有 的 计算 都 是 通过 calc 类 的 公有 
方法 实现 的 。 

这 段 脚本 虽然 看 上 去 复杂 ， 但 是 阅读 起 来 似乎 比 第 一 段 清晰 ， 至 少 通过 阅读 代码 可 以 
很 清楚 的 知道 每 一 块 的 意思 是 什么 。 可 能 有 的 读者 认为 第 一 段 脚 本 直 来 直 去 的 更 加 清晰 ， 
那 是 因为 这 两 段 脚本 实现 的 功能 都 很 简单 。 当 我 们 需要 编写 复杂 的 程序 时 ， 会 发 现 基 于 对 
象 的 编程 风格 可 以 让 脚本 的 可 读 性 大 大 提升 。 这 也 是 为 什么 我 们 要 学 习 基 于 对 象 编程 的 一 
个 重要 的 原因 。 


12.1 基于 过 程 与 基于 对 象 


12.1.1 为 什么 要 用 OOP 


基于 对 象 的 编程 让 开发 人 员 的 日 子 变 得 好 过 了 许多 。 而 正 是 因为 基于 对 象 的 编程 可 以 
把 我 们 碰 到 的 问题 变 成 一 个 一 个 更 加 细小 的 问题 从 而 使 问题 更 加 容易 理解 、 容 易 解决 。 

基于 对 象 编程 的 主要 目标 是 通过 对 象 完 成 所 有 需要 完成 的 任务 。 对 象 可 以 说 是 基于 对 
象 编程 中 最 小 、 最 基本 的 部 件 。 我 们 可 以 通过 在 对 象 间 共享 数据 来 解决 问题 。 

一 般 来 说 ， 使 用 基于 对 象 编程 的 好 处 主要 有 如 下 几 个 方面 。 

1， 可 读 性 

一 个 对 象 做 为 一 个 实体 拥有 一 系统 列 的 属性 和 方法 ， 同 时 也 可 以 和 其 他 的 对 象 互动 共 
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同 完成 某 个 任务 。 有 时 候 , 一 个 对 象 可 以 独立 存在 , 就 像 在 本 章 的 前 言 中 看 到 的 Scalc 对 象 ， 
而 有 的 对 象 则 需要 依附 于 其 他 的 对 象 。 通 常 ， 一 个 对 象 用 于 解决 一 类 具体 的 问题 ， 就 像 我 
们 定义 的 Scalec 对象, 可 以 用 来 设置 和 获取 calc 类 的 内 置 变 量 , 同时 使 用 其 内 置 的 方法 计算 
内 置 变量 的 和 差 积 商 。 如 果 在 开发 过 程 中 碰 到 了 一 系列 有 联系 的 问题 ， 那 么 可 以 尝试 定义 
一 个 解决 这 些 问 题 的 类 ， 然 后 再 通过 实例 化 该 类 来 解决 这 些 问 题 。 


2. 重用 性 


如 果 同 时 开发 儿 个 PHP 项 目 , 而 这 些 项 目 有 着 某 些 相同 点 , 那么 使 用 基于 对 象 的 编程 
可 以 方便 地 将 某 一 个 项 目 中 使 用 的 类 移植 到 另 一 个 项 目 中 ,从 而 实现 类 的 重用 。 这 样 一 来 ， 
就 不 用 在 每 一 个 项 目 中 重新 编写 脚本 了 。 


3. 扩展 性 


如 果 想 要 在 开发 的 项 目 中 方便 地 添加 新 的 方法 和 属性 ， 使 用 基于 对 象 的 编程 是 不 二 的 
选择 。 基 于 对 象 的 编程 最 核心 的 功能 就 是 扩展 性 。 可 以 重 构 对 象 来 为 其 添加 新 的 方法 或 属 
性 ， 同 时 也 可 以 继续 保持 该 对 象 的 向 下 兼容 能 力 ， 兼 容 原 有 的 脚本 。 另 外 ， 还 可 以 重新 创 
建 一 个 保留 了 其 父 本 对 象 中 必要 方法 和 属性 的 新 对 象 ， 然 后 向 其 中 添加 新 的 方法 和 属性 。 
这 种 特性 也 叫做 继承 性 ， 我 们 在 第 8 章 涉及 过 这 个 概念 。 


4. 可 维护 性 


基于 对 象 的 脚本 比 基 于 过 程 的 脚本 维护 起 来 要 容易 的 多 。 因 为 基于 对 象 的 脚本 遵循 了 
某 种 严格 的 编码 规范 ， 通 过 明白 的 逻辑 结构 实现 需要 的 功能 。 打 个 比方 ， 如 果 使 用 基于 对 
象 的 脚本 ， 那 么 可 以 方便 地 对 开发 的 脚本 进行 扩展 ， 也 可 以 将 其 分 散 成 若干 个 片段 分 发 给 
和 你 开发 同一 项 目的 其 他 开发 人 员 。 这 些 分 散 开 来 的 片段 也 可 以 成 为 独立 的 对 象 完成 特定 
的 任务 ， 大 家 互 不 影响 、 独 立 开 发 。 需 要 整合 的 时 候 ， 也 可 以 非常 容易 地 把 这 些 分 散 的 片 
段 整合 在 一 起 。 

5. 高效 性 

使 用 基于 对 象 的 编程 可 以 提升 脚本 的 执行 效率 和 简化 脚本 开发 流程 。 有 许多 既定 的 开 
发 模式 可 以 供 我 们 开发 更 好 更 高 效 的 脚本 。 

基于 上 述 原因 ， 我 们 可 以 使 用 基于 对 象 的 编程 方式 来 提升 代码 的 可 读 性 、 重 用 性 、 扩 
展 性 、 可 维护 性 和 代码 的 执行 效率 。 使 我 们 以 更 加 合理 的 开发 模式 将 任务 化 整 为 零 ， 步 步 
为 营地 将 其 解决 。 


12.1.2 ”对 象 面 面 观 


在 第 8 章 里 学 习 了 类 和 对 象 的 概念 ， 知 道 如 何 定义 一 个 类 ， 如 何 通过 实例 化 类 来 创建 
对 象 以 及 如 何 使 用 对 象 来 完成 特定 的 任务 。 在 本 小 节 里 ， 以 例 12.1 中 引用 的 calc 类 为 例 来 
解剖 一 个 对 象 。 详 细 了 解 一 下 什么 是 对 象 、 如 何 使 用 对 象 。 

【 例 12.2】 Calc 类 的 使 用 。 
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class calc { 
private $a; 
private $b; 


// 为 私有 变量 设置 初始 值 
function _construct(){ 
Sthis ->a= 5; 
$this -> b 67 


} 
// 设 置 私 有 变量 $a 和 $b; 


function setVarA($num){ 
Sthis -> a = $num; 
return S$this -> a; 


function setVarB (Snum) { 
Sthis -> b = $num; 
return S$this -> b; 


// 获 取 私有 变量 $a 和 $b; 
function getVarA(){ 
return Sthis -> a; 


function getVarB(){ 
return $this -> b; 


// 计 算 变 量 $a 和 $b 的 和 ; 

function add()1{ 
$result = Sthis -> a + Sthis -> b; 
return $result; 


上 
// 计 算 变量 $a 和 $b 的 差 ; 


function minus(){ 
$result = Sthis -> a - Sthis -> b; 
return $result; 


| 


// 计 算 变 量 $a 和 $b 的 积 ; 

function multiply()t{ 
$result = Sthis -> a * Sthis -> b; 
return $result; 


有 
// 计 算 变量 Sa 和 Sb 的 商 ; 


function divide(){ 
Sresult = Sthis -> a / Sthis -> b; 
return $result; 


} 

看 完 上 面 这 个 类 之 后 , 我 们 可 以 清楚 地 知道 , 这 个 类 有 两 个 私有 属性 和 八 个 公有 方法 。 
其 中 四 个 方法 用 来 设置 和 获取 这 两 个 私有 属性 的 值 ， 而 剩余 的 四 个 方法 则 用 来 计算 这 两 个 
私有 属性 的 和 差 积 商 。 如 果 想 要 使 用 这 个 类 ， 首 先 需 要 实例 化 这 个 类 ， 然 后 使 用 经 过 实例 
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化 的 对 象 来 使 用 上 述 八 个 公有 方法 操作 对 象 的 私有 属性 。 

那么 到 底 什么 是 对 象 呢 ? 说 白 了 ， 对 象 就 是 一 段 定义 了 若干 属性 和 方法 的 脚本 。 它 和 
数组 很 像 ， 一 个 数组 可 以 将 若干 个 值 定义 在 不 同 的 键 中 ， 而 一 个 对 象 则 拥有 若干 个 属性 和 
方法 。 它 与 数组 不 同 的 是 ， 对 象 中 的 属性 和 方法 并 不 都 是 可 以 在 实例 化 后 使 用 的 。 比 如 例 
12.2 中 的 calc 类 里 定义 的 两 个 私有 属性 在 calc 类 实例 化 后 是 不 能 直接 访问 的 。 通 常情 况 下 ， 
一 个 类 在 实例 化 后 ， 我 们 只 能 访问 其 公有 发 生 和 方法 。 

现在 , 再 仔细 地 阅读 一 下 例 12.2 中 的 calc 类 , 我 相信 这 段 脚本 在 你 眼中 会 变 得 越 来 越 
好 懂 。 如 果 你 也 使 用 这 种 风格 编写 脚本 的 话 ， 会 慢 慢 地 发 现 基 于 对 象 编写 脚本 的 好 处 。 从 
而 编写 的 脚本 也 会 变 得 越 来 越 容易 管理 和 维护 。 

在 Wordpress 的 官方 网 站 的 首页 下 方 ， 可 以 发 现 一 句 话 “Coding is Poety”《【 编 码 如 
写 诗 ) 。 只 要 施 得 其 法 ， 编 码 的 确 就 像 写 诗 一 样 。 


12.1.3 ”基于 对 象 编程 中 常用 术语 


在 本 小 节 里 ， 我 们 会 列 出 在 基于 对 象 编 程 中 经 常会 用 到 的 一 些 术 语 。 其 中 有 一 些 在 第 
8 章 里 学 习 过 ， 这 里 就 当 复 习 一 下 。 了 解 这 些 术语 以 及 它们 背后 的 意义 对 于 我 们 继续 学 习 
基于 对 象 的 编程 是 十 分 有 益 的 。 
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类 可 以 被 看 做 是 对 象 的 模板 ， 类 定义 了 对 象 可 以 使 用 的 方法 和 属性 。 基 于 某 个 类 的 对 
象 只 能 在 类 规定 的 范围 内 活动 和 与 其 他 的 对 象 互动 。 每 当 创建 了 一 个 基于 某 个 类 的 对 象 ， 
这 个 对 象 就 成 为 这 个 类 的 实例 。 

2. 属性 


属性 是 在 类 中 定义 的 ， 用 来 存放 一 些 信息 的 变量 。 值 得 注意 的 是 ， 属 性 只 能 在 类 中 定 
义 ， 并 被 基于 类 的 对 象 或 子 类 使 用 。 在 一 个 类 中 定义 的 属性 可 以 有 三 种 适用 范围 ， 一 些 只 
能 在 定义 它们 的 类 中 使 用 ， 称 为 私有 属性 ， 一些 可 以 在 定义 它们 的 类 以 及 继承 自 该 类 的 子 
类 中 使 用 ， 称 为 保护 属性 ; 还 有 一 些 则 可 以 被 所 有 人 使 用 ， 称 为 公有 属性 。 在 类 的 方法 中 
定义 的 变量 不 是 类 的 属性 ， 这 些 变量 只 能 在 这 些 函 数 中 使 用 。 

3. 方法 


方法 是 在 类 中 定义 的 ， 用 来 执行 某 些 操作 的 函数 。 它 与 属性 类 似 ， 据 其 适用 范围 的 不 
同 ， 分 为 私有 方法 、 保 护 方法 和 公有 方法 。 

4. 封装 

所 谓 封装 是 将 某 脚本 (方法 ) 和 使 用 该 脚本 (方法 ) 操控 的 数据 绑 定 在 一 起 的 一 种 机 
制 。 把 它们 封装 在 一 起 ， 可 以 有 效 地 避免 它们 受到 外 界 的 干扰 。 把 脚本 (方法 ) 和 受 脚本 
(方法 ) 操控 的 数据 绑 定 到 类 的 过 程 称 为 封装 。 类 的 封装 性 使 得 开发 人 员 不 用 再 担心 类 中 各 
方法 的 实现 原理 和 过 程 。 
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5. 多 态 


对 象 的 状态 是 多 变 的 。 一 个 对 象 相对 于 同一 个 类 的 另 一 个 对 象 来 说 ， 它 们 拥有 的 属性 
和 方法 虽然 相同 ， 但 却 可 以 有 着 不 同 的 状态 。 另 外 ， 一 个 类 可 以 派生 出 若干 个 子 类 ， 这 些 
子 类 在 保留 了 父 对 象 的 某 些 属性 和 方法 的 同时 ， 也 可 以 定义 一 些 新 的 方法 和 属性 ， 甚 至 于 
完全 改写 父 类 中 的 某 些 已 有 的 方法 ， 这 一 过 程 称 为 类 多 态 化 。 


6. 继承 


继承 是 类 多 态 化 的 诸多 方法 中 最 典型 的 一 种 。 当 定义 一 个 继承 自 另 一 个 类 的 子 类 时 ， 
子 类 可 以 继承 父 类 的 所 有 方法 和 属性 ， 也 可 以 改写 其 中 的 某 些 方法 和 属性 ， 还 可 以 添加 一 
些 新 的 方法 和 属性 。 


7. 耦合 


耦合 指 的 是 类 之 间 的 依赖 程度 。 使 用 轻 耦 合 类 间架 构 的 脚本 相 较 于 使 用 重 耦 合 类 间架 
构 的 脚本 可 用 性 要 更 高 。 在 下 一 节 里 ， 我 们 会 了 解 到 关于 耦合 这 个 概念 的 更 多 细节 。 


8. 模式 


这 里 的 模式 特 指 设计 模式 , 在 基于 对 象 的 编程 中 , 它 常 常用 来 解决 一 系列 相似 的 问题 。 
使 用 特定 的 设计 模式 ， 可 以 用 极 少 的 脚本 改写 提升 脚本 的 执行 效率 。 当 然 ， 特 定 的 设计 模 
式 是 把 双 刃 剑 ， 如 果 在 不 需要 它 的 地 方 使 用 了 它 ， 则 会 降低 脚本 的 执行 效率 。 


9. 子 类 


子 类 是 指 从 某 个 类 中 派生 出 来 的 类 。 通 常 拥有 父 类 的 某 些 属性 和 方法 ， 同 时 也 有 一 些 
自 有 的 属性 和 方法 ， 还 有 可 能 拥有 一 些 改写 过 的 父 类 的 属性 和 方法 。 


10. 父 类 
父 类 是 指派 生出 子 类 的 类 。 
11. 实例 


当 创 建 一 个 基于 某 个 类 的 对 象 时 ， 就 实例 化 了 这 个 类 。 而 这 个 被 创建 的 对 象 就 是 该 类 
的 一 个 实例 。 


12.1.4 ”基于 对 象 编程 的 编码 规范 


在 本 章 的 代码 里 ， 我 们 会 遵循 一 些 编码 规范 。 这 些 规范 虽然 不 甚 严格， 但 也 足以 使 脚 
本 可 读 和 可 维护 。 另 外 ， 在 遵循 这 些 规范 的 同时 ， 也 可 以 避免 引入 一 些 见 余 的 对 象 ， 从 而 
提升 脚本 的 执行 效率 。 

现在 一 起 来 看 看 这 些 规范 : 

口 在 一 个 php 文件 里 只 定义 一 个 类 。 在 这 个 类 之 外 ， 不 使 用 基于 过 程 的 编码 风格 。 

口 类 和 存放 类 的 文件 要 有 合适 的 名 字 。 比 如 在 例 12.2 中 定义 的 类 ， 就 应 该 放 在 名 为 
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class.calc php 文件 中 并 放 在 名 为 class 的 文件 夹 下 。 这 样 做 的 好 处 是 ， 可 以 很 方便 
地 在 一 堆 文 件 中 找到 想 要 找 的 文件 ， 而 不 用 费时 费力 地 一 个 文件 一 个 文件 地 查看 。 
在 为 类 的 文件 命名 时 ， 都 使 用 小 写字 母 。 

与 命令 一 般 类 和 类 文件 一 样 ， 对 于 存放 接口 类 的 文件 定义 成 interface name.php， 
保存 在 名 为 interface 的 文件 夹 下 ; 对 于 存放 抽象 类 的 文件 则 定义 成 
abstractname.php， 保 存在 名 为 abstract 的 文件 夹 下 ; 对 于 存放 Final 类 的 文件 则 定 
义 成 final.name.php， 保 存在 名 为 final 的 文件 夹 下 。 
用 驼峰 命令 法 来 命名 类 ， 如 果 一 个 类 的 名 字 只 含有 一 个 英文 单词 ， 则 全 部 小 写 ， 
如 calc; 如 果 一 个 类 的 名 字 中 含有 多 个 英文 单词 ， 则 从 第 二 个 单词 开始 ， 每 个 单词 
的 首 字 母 大 写 ， 如 myCalc。 

在 类 中 定义 属性 和 方法 时 ， 也 请 务必 使 用 驼峰 命令 法 。 


12.2” 初 识 OOP 


在 正式 开始 编写 基于 对 象 的 代码 之 前 ， 我 们 需要 复习 一 下 在 第 8 章 里 学 习 到 的 一 些 基 


本 概念 


12:2:1 


， 包 括 类 、 对 象 、 继 承 、 改 写 、 修 饰 和 魔术 方法 等 。 


类 和 对 象 


在 第 8 章 学 习 了 类 和 对 象 的 概念 。 在 这 一 小 节 里 ， 我 们 来 复习 与 类 和 对 象 相关 的 一 些 
基本 概念 。 


首先 还 是 来 看 一 段 脚 本 。 


【 例 12.3】 创建 名 为 notes 的 类 (\\class.notes.php) 。 
<?php 
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// 创 建 一 个 名 为 notes 的 类 
class notes { 
// 定 义 一 些 属性 
private $noteDate; 
private S$weather; 
private $author; 
private $filename; 


function _ construct ($author, $weather){ 
// 为 私有 属性 赋 初 始 值 
Sthis -> setNoteDate(date('Y 年 m 月 d 日 ')); 
Sthis -> setWeather (S$weather); 
Sthis -> setAuthor ($author); 
Sthis -> setFileName (Sthis -> author); 


// 初 始 化 存储 日 志 的 文本 文件 
Sthis -> initStorage () 7 
} 


// 配 置 存储 日 志 的 文本 文件 


private function initStorage(){ 
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$filename = Sthis -> filename; 
if(!file exists(Sftilename))1{ 
$fh = fopen ($filename, 'w'); 
fwrite ($fh, "-—---start-————\r\n"); 
fclose ($fh); 


’ 
// 定 义 设置 日 志 作 者 、 天 气 和 日 期 的 方法 


private function setFileName ($author){ 


$this -> filename = 'file/'.base64 encode($author).'.notes'; 


private function setAuthor ($author){ 
Sthis -> author = $author; 


private function setWeather ($weather) { 
Sthis -> weather = S$weather; 


private function setNoteDate ($noteDate) { 
Sthis -> noteDate = $noteDate; 


// 定 义 获 取 日 志 作者 、 天 气 和 日 期 的 方法 
protected function getFileName(){ 
return Sthis -> filename; 


function getAuthor(){ 
return Sthis -> author; 


function getWeather(){ 
return S$this -> weather; 


function getNoteDate(){ 
return Sthis -> noteDate; 


// 定 义 获取 所 有 日 志和 指定 日 志 的 方法 
function getNotes(){ 
$filename = $this -> filename; 


$fh = fopen ($filename, 'r'); 

$i = 0; 

while (!feof(Sfth)) { 
$notes[$i] = fgets ($fh); 
$it++? 

} 

fclose ($fh); 


// 移 除 首尾 两 个 元 素 
array_pop ($notes); 


array shift ($notes); 


return S$notes; 
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protected function getNote (SnoteIndex)1{ 
Snotes = $this -> getNotes () 


return $notes[$noteIndex]; 


} 


// 定 义 格式 化 输出 所 有 日 志 的 方法 
function displayNotes(){ 
Snotes = $this -> getNotes(); 


// 输 出 日 志 记 录 
if(count ($notes) > 0){ 
foreach ($notes as SnoteId => $note) { 
Snote = explode('|', $note); 
echo '<div class="note"><table> 
<th class="center"> 
<td>#"'. (SnoteId+1) . "</td> 
<td>' .$note[0].'</td> 
<td>' .$note[1].'</td> 
<td>' .S$note [2] .'</td> 
<td> 
<a href="?modNote=' .SnoteId.'"> 修 改 </a> | 
<a href="?delNote=' .$noteId.'"> 删 除 </a> 
</td> 
</th> 
<tr class="indent"> 
<td colspan="4">'.$note[3].'</td> 
</tr> 
</table> 
</div>'; 
} 
} else { 
echo '<div class="note"><p class="center"> 暂 无 日 志 记 录 。 
</p></div>"'; 


} 


// 定 义 添加 日 志 的 方法 
function addNote ($note) { 
// 将 作者 、 日 期 和 天 气 信 息 整 合 到 日 志 中 
$note = implode('|', array($this -> getAuthor(), 
Sthis -> getWeather (), 
Sthis -> getNoteDate(), 
Snote) ) 7 


// 获 取 存 储 日 志 的 文件 名 ， 并 在 其 尾部 追加 新 日 志 
$filename = S$this -> getFileName(); 
$fh = fopen ($filename, ‘'at+'); 


if (fgets ($fh) == '----start--——'){ 
fwrite ($fh, $note); 
} else { 


fwrite ($fh, $note."\r\n"); 
fclose ($fh); 
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在 上 面 这 段 脚本 中 以 “class notes” 开 始 ， 定 义 了 一 个 名 为 notes 的 类 。 紧 接着 定义 了 
一 系列 属性 ， 它 们 分 别 是 写 日 志 的 时 间 〈$noteDate) 、 日 志 的 作者 〈$author) 、 写 日 志 时 
的 天 气 〈$weather) 以 及 用 来 存储 日 志 的 文本 文件 名 〈$filename) 。 在 定义 完 notes 类 的 属 
性 后 ， 在 _construct 魔术 方法 中 使 用 私有 方法 setAuthor、setWeather、setNoteDate 和 
$setFileName 为 这 四 个 私有 属性 赋 了 值 。 这 就 意味 着 这 四 个 私有 属性 的 数据 类 型 也 就 确定 
了 ， 它 们 都 是 字符 串 型 的 变量 。 

在 魔术 方法 中 ， 还 初始 化 了 用 来 存储 用 户 日 志 的 文本 文件 。 该 文件 的 文件 名 是 用 户 姓 
名 加 密 后 生成 的 一 个 字符 串 ， 扩 展 名 为 “.note”; 若 该 文件 不 存在 ， 则 创建 该 文件 。 

接 下 来 ， 我 们 定义 了 用 来 设置 和 获取 私有 属性 的 八 个 方法 。 设 置 私有 属性 的 方法 有 
setAuthor、setWeather、setNoteDate 和 setFileName， 这 四 个 方法 均 为 私有 方法 ， 要 求 提 供 
一 个 参数 且 都 没有 返回 值 。 类 似 这 样 的 方法 名 以 “set” 打 头 的 方法 可 以 叫做 “setter ”方法 。 
获取 私有 属性 的 方法 有 getAuthor、getWeather、getNoteDate 和 getFileName， 这 四 个 方法 
中 除了 getFileName 为 受 保护 的 方法 外 ， 其 他 三 个 方法 为 公有 方法 。 这 四 个 方法 都 没有 参 
数 要 求 且 都 返回 对 应 私有 属性 的 值 。 类 似 这 样 的 方法 名 以 “get” 打 头 的 方法 叫做 “getter” 
方法 。 

然后 ， 我 们 定义 了 获取 所 有 日 志和 指定 日 志 的 方法 。 它 们 分 别 是 getNotes 和 getNote， 
前 者 为 公有 方法 ， 后 者 为 受 保护 的 方法 ;前 者 读 取 存 储 日 志 的 文件 ， 并 将 读 取 到 的 内 容 存 
入 一 个 数组 中 ， 然 后 返回 数组 ， 该 方法 不 要 求 提供 任何 参数 ， 后 者 则 引用 了 getNotes 方法 
将 日 志文 件 中 的 内 容 存 入 到 一 个 数组 中 ， 然 后 在 数组 用 指定 的 索引 获取 对 应 的 日 志 记录 ， 
该 参数 要 求 提供 指定 日 志 记录 的 索引 值 。 另 外 ， 我 们 还 定义 了 用 于 在 页 面 上 展示 日 志 列 表 
的 displayNotes 方法 ， 该 方法 为 公有 方法 ， 不 要 求 提 供 参数 ， 也 不 返回 任何 值 ， 它 只 会 按 
照 既 定 的 格式 向 页 面 输出 日 志 记录 。 

最 后 我 们 定义 了 添加 日 志 的 方法 addNote， 均 为 公有 方法 。 添 加 日 志 的 方法 要 求 提供 
一 个 参数 ， 且 该 方法 不 返回 任何 值 ， 而 是 直接 将 操作 完成 后 的 日 志 记 录 重 新 写 到 文本 文 
件 中 。 

值得 注意 的 是 , 我 们 在 类 中 定义 方法 的 时 候 ， 使 用 了 Sthis 变量 来 访问 类 中 定义 的 属性 
和 方法 。 该 变量 代 指 类 本 身 ， 而 “->” 符 号 则 表示 “访问 ”。 以 setAuthor 方法 为 例 : 


private function setRuthor (Sauthor){ 
Sthis -> author = $author; 


} 


在 定义 setAuthor 方法 时 ， 我 们 使 用 $this -> author 来 访问 notes 类 的 $author 属性 。 由 
于 $author 属性 是 notes 类 的 私有 属性 ， 因 此 ， 该 属性 只 能 在 类 中 通过 $this 变量 引用 。 同 样 
也 ， 如 果 需 要 在 类 中 引入 setAuthor 方法 ， 我 们 也 可 以 通过 $this 变量 ， 如 $this->setAuthor。 
这 里 ，setAuthor 为 notes 类 的 私有 变量 ， 只 能 在 类 中 通过 $this 变量 访问 。 

还 有 一 点 需要 注意 的 是 , 在 类 中 定义 的 方法 , 除了 setter 和 getter 类 方法 之 外 ， 其 他 的 
方法 无 论 其 状态 如 何 , 都 不 建议 直接 访问 类 的 私有 属性 , 而 要 通过 setter 和 getter 类 方法 来 
间接 获取 。 这 样 ,一旦 修改 了 私有 属性 的 名 称 ， 就 只 用 修改 setter 和 getter 类 中 引用 的 相应 
的 私有 属性 ， 否 则 ， 可 能 得 在 整个 类 的 脚本 中 ， 修 改 散 落 在 各 处 的 私有 属性 名 称 。 

现在 通过 实例 化 notes 类 来 创建 一 个 基于 notes 类 的 应 用 程序 。 
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【 例 12.4】 创建 一 个 基于 notes 类 的 应 用 程序 (\Ex12-4.php)。 


<?php 
include once('class/class.notes.php'); 


// 定 义 Notes 的 使 用 者 和 当前 天 气 情况 
Sauthor = ' 张 晓 明 '; 


Sweather = ' 晴 '; 


// 初 始 化 notes 类 


$nts = new notes ($author, $weather); 


2 
<!DOCTYPE html> 
<htm1l lang="en"> 
<head> 
<title>Notes</title> 
<meta charset="UTF-8"> 
<link href="css/Ex12-04.css" rel="stylesheet"> 
</head> 
<body> 
<div class="header"> 
<h2> 我 的 日 志 </h2> 


<hr /> 
</div> 
<div class="notebox"> 
<table> 
<tr> 
<td class="triple indent"> 作 者 : <?php echo S$nts-> 
getRuthor () ; ?></td> 
<td class="triple center"> 日 期 : <?php echo $nts-> 
getNoteDate(); ?></td> 
<td class="triple center"> 天 气 : <?php echo $nts-> 
getWeather () ; ?></td> 
</tr> 
攻关 之 
<td colspan="3"> 
<form class="center" action="?addNote" method="post"> 
<textarea rows="7" cols="100" name="note"> 请 在 
此 输入 日 志 。</textarea> 
<input type="submit" value=" 我 写 完了 "> 
</form> 
</td> 
</tr> 
</table> 
</div> 
<div class="notelist" id="notelist"> 
<?php 


// 当 用 户 添加 新 日 志 时 执行 如 下 脚本 

if(isset($ REQUEST['addNote']))t{ 
$note = $ REQUEST['note']; 
Snts -> addNote ($note); 


S$notes = $nts -> displayNotes(); 
} else { 
S$notes = $nts -> displayNotes(); 


} 


BS 
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</div> 
</body> 

</html> 

上 面 这 段 脚本 中 ， 实 例 化 了 notes 类 ， 并 在 页 面 上 引用 了 实例 化 后 的 nts 对 象 的 
getAuthor、getNoteDate 和 getWeather 方法 获取 了 nts 对 象 相 关 私 有 属性 的 值 。 

接着 我 们 引用 了 nts 对 象 的 getNotes 方法 获取 了 存储 在 文本 文件 中 的 日 志 记录 。 若 文 
本 文件 不 存在 ，Snts 对 象 会 自动 创建 该 文件 。 

当 用 户 提 交 表单 后 ， 我 们 引用 了 nts 对 象 的 addNote 方法 将 用 户 提交 的 内 容 存储 到 指 
定 的 文本 文件 中 。 上 面 这 段 脚本 最 终 运 行 的 样子 如 图 12-1 所 示 。 


€ 3 CC Dwww.greatwalltea/chapterl2/Ex12-04.php 
[3 Suggested Sites [ Web Slice Gallery © Imported From IE 


我 的 日 志 


日 期: 2013 年 10 月 07 日 


1 张晓明 2013 年 10 月 06 日 tik | 
今天 天 气 无 比 博 季 ,心情 大 好 。 品 然 工作 还 是 忙碌 ， 但 心 却 漳 焰 。 

#2 张晓明 2013 年 10 月 07 日 ek | 
今天 雾 芥 终于 散 去 ， 天 空 虽然 还 不 是 十 分 的 干 兆 ， 但 也 算 怕人 了 。 


图 12-1 创建 一 个 基于 notes 类 的 应 用 程序 


12.2.2 ”类 的 扩展 和 改写 


基于 对 象 的 编程 最 令 人 称道 的 一 点 就 是 其 扩展 性 和 继承 性 。 我 们 可 以 通过 扩展 一 个 类 
而 获得 一 个 新 类 ， 该 新 类 继承 了 原 有 类 的 所 有 属性 和 方法 ， 是 原 有 类 的 子 类 。 在 上 一 小 节 
中 定义 的 类 只 有 添加 日 志 的 功能 ， 修 改 和 删除 的 功能 还 没有 实现 。 在 本 小 节 里 ， 我 们 来 扩 
展 一 下 原 有 的 notes 类 ， 创 建 一 个 powerNotes 类 ， 并 为 powerNotes 类 添加 两 个 新 方法 
modNotes 和 delNotes。 

【 例 12.5】 创建 一 个 扩展 notes 类 的 powerNotes 类 (\\class.powernotes.php) 。 

<?php 

// 导 入 类 文件 


include once('class.notes.php'); 


// 创 建 一 个 扩展 于 notes 类 的 新 类 ， 名 为 powernotes 
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class PowerNotes extends notes { 


// 为 powerNotes 类 添加 两 个 新 的 方法 


function modNote ($noteIndex, S$newNote){ 


// 获 取 指 定 记录 


$note = $this -> getNote ($noteIndex); 


// 蔡 换 现 有 记录 中 的 日 志 内 容 

$note = explode('|', $note); 
Snote[3] = $newNote; 

// 重 新 组 合 指定 日 志 

$note = implode('|', $note); 
// 将 指定 日 志 还 原 到 数组 中 

$notes = $this -> getNotes(); 
Snotes [SnoteIndex] = $note; 
// 获 取 文件 名 


$filename = $this -> getFileName(); 


// 删 除 之 前 的 文件 


unlink ($filename); 


// 新 建 一 个 同名 的 文件 

$fh = fopen ($filename, 'w'); 
fwrite ($fh, "----start-—--\r\n"); 
fclose ($fh); 


// 遍 有 历 现 有 日 志 记录 ， 并 将 其 写 入 新 建 的 同名 文件 中 
foreach ($notes as S$key => $value) { 
if(S$key == $noteIndex){ 
Sthis -> addNote ($value); 
} else { 
SnoteLength = mb_ strlen($value, 'utf8°'); 
$value = mb_substr ($value, 0, $noteLength-2, 'utf8'); 
Sthis -> addNote ($value); 


} 


function delNote ($noteIndex){ 
// 获 取 日 志 记 录 
Snotes = $this -> getNotes(); 


// 删 除 指定 的 记录 


unset (Snotes [SnoteIndex]) 7 


// 获 取 文 件 名 


$filename = $this -> getFileName(); 


// 删 除 之 前 的 文件 


unlink ($filename); 


// 新 建 一 个 同名 的 文件 

$fh = fopen ($filename, 'w'); 
fwrite ($fh, "--—--start-————\r\n"™); 
fclose ($fh); 
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// 遍 历 现 有 日 志 记录 ， 并 将 其 写 入 新 建 的 同名 文件 中 

foreach ($notes as S$key => $value) { 
SnoteLength = mb strlen($value, "utf8') 7 
$value = mb substr($value, 0, $noteLength-2, "utf8') 7 
Sthis -> addNote ($value); 


} 

2> 

在 定义 powerNotes 类 时 , 我 们 使 用 了 extends 关键 字 (class powerNotes extends notes ) ， 
表明 现在 定义 的 powerNotes 类 是 notes 类 的 子 类 ， 继 承 了 notes 类 的 所 有 公有 和 受 保护 的 
属性 和 方法 。 

需要 注意 的 是 ， 子 类 不 能 直接 使 用 父 类 的 私有 属性 和 方法 。 

接着 为 powerNotes 类 定义 了 两 个 新 的 方法 modNotes 和 delNotes。 这 两 个 方法 都 是 公 
有 方法 ， 前 者 需要 提供 修改 的 日 志 记 录 的 索引 和 修改 后 的 日 志 记录 ， 后 者 要 求 提 供需 要 删 
除 的 日 志 记录 的 索引 。 这 两 个 方法 都 不 会 返回 任何 值 。 

编写 这 两 个 方法 的 思路 是 ， 从 现 有 的 文本 文件 中 读 取 已 有 的 日 志 记 录 ， 并 将 其 存放 到 

-个 数组 中 。 然 后 通过 参数 传递 过 来 的 日 志 记 录 索 引 在 数组 中 查找 相应 的 日 志 记录 。 在 修 

改 或 删除 指定 的 日 志 记录 对 应 的 数组 元 素 后 ， 删 除 现 有 文件 ， 并 创建 一 个 同名 的 新 文件 。 
然后 将 修改 或 删除 了 指定 日 志 记 录 对 应 的 数组 元 素 的 数组 逐条 写 入 新 文件 中 。 再 把 修改 数 
组 写 入 到 新 文件 的 过 程 中 ， 使 用 了 继承 自 notes 类 的 addNotes 方法 。 

下 面 我 们 把 powerNotes 类 实例 化 ， 创 建 一 个 名 为 Spn 的 对 象 ， 然 后 引用 addNotes、 
modNotes 和 delNotes 方法 执行 创建 、 修 改 和 删除 日 志 的 操作 ， 看 看 会 有 什么 结果 。 

【 例 12.6】 创建 、 修 改 和 删除 日 志 的 操作 〈\NEx12-06.php) 。 


<?php 
include _ once ('class/class.Powernotes.Php') 


$author = “" 白 秋明 "; 
Sweather = “" 睛 "; 


$pn = new powerNotes ($author, S$weather); 
2 
<!DOCTYPE html> 
<html> 
<head> 
<title> 添 加 、 修 改 和 删除 日 志 记录 </title> 
<meta charset="UTF-8" > 
<style> 
pre { 
font-size: 1.2em; 


} 


</style> 
</head> 
<body> 
<?php 
// 在 执行 操作 前 输出 现 有 记录 


echo '<pre>';var dump($pn -> getNotes ()) ;echo '</pre>'; 
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2> 


<form action="?addNote" method="post"> 
<input type="submit"” value=" 添 加 日 志 "> 
</form> 


<form action="?modNote" method="post"> 
<input type="submit" value=" 修 改 日 志 "> 
</form> 


<form action="?delNote" method="post"> 
<input type="submit"” value=" 删 除 日 志 "> 
</form> 
</body> 
</html> 


<?php 
if(isset($ REQUEST['addNote']))t{ 
$pn -> addNote ("中 秋 节 快乐 1") ; 
// 在 执行 操作 后 再 次 输出 现 有 记录 ， 看 看 是 否 添加 了 日 志 记录 
echo '<pre>';var dump($pn -> getNotes () ) :echo '</pre>'; 
} 


if(isset($ REQUEST['modNote'])){ 
$pn -> modNote (2,， "国庆 节 快 乐 1") ; 


// 在 执行 操作 后 青 次 输出 现 有 记录 ， 看 看 是 否 修 改 了 指定 的 日 志 记录 
echo '<pre>';var dump($pn -> getNotes());echo '</pre>'; 
} 


if(isset($ REQUEST['delNote'])){ 
$pn -> delNote(2); 


// 在 执行 操作 后 再 次 输出 现 有 记录 ， 看 看 是 否 删除 了 指定 的 日 志 记录 
echo '<pre>';var dump($pn -> getNotes () ) ;echo '</pre>'; 


和 


在 例 12.6 的 脚本 中 ,首先 导入 了 class.powernotes.php 文件 , 然后 通过 实例 化 powerNotes 
类 创建 了 名 为 $pn 的 实例 。 
接着 ， 我 们 使 用 powerNotes 类 继承 自 notes 类 的 getNotes 方法 输出 了 日 志文 件 中 的 
内 容 。 
在 该 页 面 的 HTML 部 分 定义 了 三 张 表单 ， 分 别 用 来 添加 、 修 改 和 删除 日 志 。 添加 的 日 
志 内 容 为 “中 秋 节 快乐 ”， 修 改 后 的 日 志 内 容 为 “国庆 节 快 乐 ”。 在 执行 了 这 三 类 操作 中 
的 任意 一 种 后 , 系统 都 会 再 次 输出 日 志文 件 的 内 容 , 供 我 们 比 对 操作 前 后 日 志文 件 的 变化 。 
需要 注意 的 是 ， 在 修改 和 删除 日 志 的 两 个 方法 中 指定 的 日 志 记录 索引 为 2， 因 此 要 添 
加 至 少 三 条 日 志 记 录 。 
现在 打开 页 面 ， 连 续 单 击 三 次 “添加 日 志 ” 按 钮 。 页 面 的 状态 如 下 : 
观察 图 12-2， 可 以 发 现 最 后 一 次 单 击 “ 添 加 日 志 ” 按 钮 前 日 志文 件 中 有 两 条 记录 ， 而 
其 后 ， 日 志 记 录 增 加 到 了 三 条 。 我 们 的 addNotes 方法 奏效 了 。 
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€ 3 CC 口 wwwgreatwalltea/chapterl2/Exl2-06php?addNote 
回 Suggested Sites 口 Web Sfice Gallery DD Imported From 正 


array(2) { 


[e]=> 
“string(59)“ 白 秋明 | 晴 |2613 年 19 月 67 日 | 中 秋 节 快乐 ! 


[1]=> 
string(59)“ 白 秋明 | 晴 |2913 年 19 月 97 日 | 中 秋 节 快乐! 


array(3) { 
[e]=> 
string(59) “ 白 秋 明 | 畏 |2613 年 16 月 67 日 | 中 秋 节 快乐 ! 


[1]=> 
string(58)“ 白 秋明 | 晴 |2913 年 16 月 97 日 | 中 秋 节 快乐 ! 


[2]-> 
string(59) “ 白 秋 明 | 畏 |2613 年 16 月 97 日 | 中 秋 节 快乐 ! 


12-2 添加 日 志 记 录 


现在 我 们 再 单 击 “ 修 改 日 志 ” 按 钮 ,把 索引 号 为 2 的 日 志 记 录 的 内 容 由 “中 秋 节 快乐 ” 


修改 为 “国庆 节 快 乐 ”。 修 改 后 的 页 面 如 图 12-3 所 示 。 


口 添加 、 修 改 和 出 除 日 志 i: x 3 

€ 3 CC Dwwwgreatwalltea/chapterl2/Ex12-06.php?modNote 
[S) Suggested Sites 站 Web Slice Gallery OD Imported From IE 

array(3) { 


[9]=> 
。5tring(59)“ 白 秋明 | 晴 |2813 年 19 月 97 日 | 中 秋 节 快乐! 


[1]=> 
string(59)“ 白 秋明 | 晴 |2913 年 19 月 97 日 | 中 秋 节 快乐! 


[2]=> 
string(59) “ 白 秋明 | 晴 12913 年 16 月 67 日 | 中 秋 节 快乐 ! 


array(3) { 
[e]=> 
。string(82) “ 白 秋 明 | 晴 12813 年 16 月 97 日 | 白 秋明 | 晴 12913 年 16 月 7 日 | 中 秋 节 快乐 ! 


[1]=> 
。string(82) “ 白 秋明 | 畏 12613 年 16 月 67 日 | 白 秋 明 | 畏 12613 年 16 月 67 日 | 中 秋 节 快乐 ! 


[2]=> 
string(82) “ 白 秋明 | 晴 12613 年 1 月 97 日 | 白 秋明 | 晴 |2913 年 19 月 97 日 | 国庆 节 快乐 1 


图 12-3 ”修改 日 志 记 录 


在 修改 完成 后 ， 发 现 索 引 为 2 的 日 志 记 录 的 内 容 的 确 由 “中 秋 节 快乐 ”修改 成 了 “ 国 
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庆 节 快乐 ”。 但 是 为 什么 每 条 ee 了 呢 ? 
再 单 击 “删除 日 志 ” 按 钮 后 ， 每 条 日 志 记录 的 内 容 变 得 更 加 长 了 。 
这 是 什么 原因 造成 的 呢 ? 原来 ， ems 和 delNotes 中 都 引用 了 addNotes 方法 ， 
而 addNotes 方法 中 拼接 日 志 记录 的 时 候 会 把 日 志 的 作者 、 日 志 完成 的 时 间 和 当时 的 天 气 情 
况 连 同 添加 的 日 志 一 起 添加 到 a et Mb 
在 modNotes 和 delNotes 中 使 用 addNotes 时 , 代入 到 addNotes 中 的 参数 不 单单 是 日 志 记 录 
的 内 容 ， 而 是 包括 了 日 志 作 者 、 日 志 完 成 的 时 间 和 当时 的 天 气 情况 等 信息 的 完整 的 日 志 记 
录 。 因 而 出 现 了 如 图 12-3 所 示 的 情况 。 

修正 这 个 Bug 有 两 个 思路 ， 一 个 是 修改 在 定义 modNotes 和 DelNotes 方法 时 代入 到 
addNotes 方法 中 的 参数 ， 另 一 个 就 是 改写 addNotes 方法 。 

在 基于 对 象 的 编程 中 ， 类 间 的 继承 和 改写 是 常 有 的 事 。 现 在 就 在 powerNotes 类 中 ， 改 
写 一 下 addNotes 方法 。 

【 例 12.7】 在 powerNotes 类 中 改写 其 继承 自 notes 类 的 addNotes 方法 〈\Nclass. 
powernotes.php) 。 


<?php 
// 导 入 类 文件 


include once('class.notes.php'); 


// 创 建 一 个 扩展 于 notes 类 的 新 类 ， 名 为 powernotes 
class powerNotes extends notes { 
// 为 powerNotes 类 添加 两 个 新 的 方法 


function modNote ($noteIndex, S$newNote){ 


// 响 历 现 有 日 志 志 记录 ， 并 将 其 写 入 新 建 的 同名 文件 中 
foreach ($notes as $key => $value) { 
if($key == $noteIndex){ 
$this -> addNote ($value, 'mod'); 
} else { 
SnoteLength = mb strlen($value, "utf8')7 
$value = mb substr($value, 0, $noteLength-2, "utf8'); 
$this -> addNote ($value, 'mod'); 


} 


function delNote (SnoteIndex) { 


// 遍 历 现 有 日 志 记录 ， 并 将 其 写 入 新 建 的 同名 文件 中 

foreach ($notes as $key => $value) { 
SnoteLength = mb strlen($value, "utf8') 7 
$value = mb_substr($value, 0, $noteLength-2, "utf8') 7 
$this -> addNote($value, 'del'); 


. 


// 改 写 继承 自 notes 类 的 addNotes 方法 
function addNote ($note, $action = 'add'){ 


// 将 作者 、 日 期 和 天 气 信 息 整 合 到 日 志 
if ($action == 'add') { 
Snote = implode('|', array($this -> getAuthor(), 
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Sthis -> getWeather (), 
Sthis -> getNoteDate(), 
Snote) ) 

} 


// 获 取 存 储 日 志 的 文件 名 ， 并 在 其 尾部 追加 新 日 志 
$filename = $this -> getFileName(); 
$fh = fopen ($filename, 'a+'); 


i (fgets(SEh) — 1 "===start===="){ 
fwrite ($fh, $note); 
} else { 


fwrite ($fh, $note."\r\n"); 
} 


fclose ($fh); 


} 

?> 

所 谓 改写 ， 其 实 就 是 把 父 类 中 的 方法 或 属性 再 定义 一 遍 。 在 上 面 这 段 脚本 中 ， 我 们 在 
继承 自 父 类 的 addNotes 方法 中 添加 一 个 名 为 $action 的 参数 , 并 且 这 个 参数 是 可 选 参数 ,如 
果 不 指定 的 话 ， 那 么 这 个 参数 就 会 使 用 默认 值 “add”。 而 只 有 当 $action 参数 值 为 “add” 
时 ， 系 统 才 会 拼接 日 志 内 容 和 日 志 作者 等 信息 。 接 下 来 ， 需 要 在 modNotes 和 delNotes 两 
个 方法 中 引入 addNotes 时 将 $action 参数 分 别 指定 为 “mod” 和 “del”( 其 实 这 里 的 值 是 什 
么 并 不 重要 ， 只 要 不 是 “add” 就 好 ) 。 

现在 再 来 运行 一 下 例 12.6 中 的 脚本 。 添 加 三 条 记录 ， 然 后 修改 和 删除 第 三 条 记录 。 你 
会 发 现 ， 刚 才 发 现 的 Bug 已 经 成 功 地 清除 掉 了 。 

用 改写 类 的 方法 是 不 是 要 比 修改 在 定义 modNotes 和 DelNotes 方法 时 代入 到 addNotes 
方法 中 的 参数 容易 的 多 呢 ? 


12.2.3 ”修饰 词 


知道 如 何 改写 继承 自 父 类 的 方法 和 属性 后 ， 我 们 就 可 以 对 所 有 继承 自 父 类 公有 的 和 受 
保护 的 方法 与 属性 进行 改写 。 不 过 对 于 那些 可 以 在 子 类 中 使 用 ， 却 不 建议 在 子 类 中 改写 的 
方法 和 属性 ， 我 们 就 得 在 父 类 中 定义 这 些 方 法 和 属性 时 使 用 final 修饰 词 。 

使 用 PHP 基于 对 象 的 编程 中 经 常用 到 的 修饰 词 有 如 下 儿 种 。 

(1) public 

使 用 public 修饰 的 方法 和 属性 称 为 公有 方法 和 属性 ， 可 以 在 定义 该 方法 和 属性 的 类 的 
子 类 和 实例 中 使 用 。 默 认 情 况 下 ，public 修饰 词 不 是 必需 的 ， 所 有 不 带 修饰 词 的 方法 和 属 
性 都 是 公有 的 。 

(2) Private 

使 用 private 修饰 的 方法 和 属性 称 为 私有 方法 和 属性 , 只 能 在 定义 该 方法 和 属性 的 类 中 
使 用 。 在 其 子 类 和 实例 中 ， 这 些 方 法 和 属性 都 是 不 可 见 的 。 默 认 情 况 下 ， 只 有 在 定义 某 个 
方法 或 属性 时 使 用 了 private 修饰 词 ， 该 方法 或 属性 才 是 私有 的 。 

(3) protected 

使 用 protected 修饰 的 方法 和 属性 称 为 受 保护 的 方法 和 属性 , 只 能 在 定义 该 方法 和 属性 
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的 类 及 其 子 类 中 使 用 。 对 于 定义 该 方法 和 属性 的 类 及 其 子 类 的 实例 来 说 ， 该 方法 和 属性 均 
不 可 见 。 默 认 情 况 下 ， 只 有 在 定义 某 个 方法 或 属性 时 使 用 了 protected 修饰 词 ， 该 方法 或 属 
性 才 是 受 保护 的 。 

(4) static 

使 用 static 修饰 的 方法 和 属性 称 为 静态 方法 和 静态 属性 ， 它 在 基于 对 象 的 编程 中 至 关 
重要 。 对 于 在 某 个 类 中 定义 的 静态 方法 和 静态 属性 ， 我 们 可 以 在 该 类 没有 被 实例 化 的 情况 
下 直接 调用 这 些 方法 和 属性 。 一 个 静态 方法 和 属性 ， 其 状态 对 于 定义 该 方法 和 属性 的 类 及 
该 类 的 实例 来 说 都 是 一 致 的 。 举 个 例子 ， 当 改变 了 某 静 态 属 性 的 值 ， 那 么 基于 定义 这 个 静 
态 属性 的 类 的 所 有 实例 都 会 使 用 该 属性 改变 后 的 值 来 代入 运算 。 

(5) final 

使 用 final 修饰 的 方法 和 属性 称 为 死 方法 和 死 属 性 , 这 些 方法 或 属性 无 论 在 何 时 都 是 不 
可 以 被 改写 的 。 在 final 修饰 词 前 还 可 以 添加 上 述 三 类 修饰 词 ， 共 同 决定 它们 修饰 的 方法 或 
属性 的 适用 范围 。 比 如 ， 如 果 我 们 使 用 private final 来 修饰 一 个 方法 ， 那 么 这 个 方法 只 能 在 
定义 它 的 类 中 使 用 ， 且 不 可 被 改写 ; 如 果 我 们 使 用 protected final 来 修饰 一 个 方法 ， 那 么 这 
个 方法 只 能 在 定义 它 的 类 及 其 子 类 中 使 用 ， 且 在 该 类 及 其 子 类 中 均 不 可 被 改写 。 

另外 ， 与 上 述 三 种 修饰 词 不 同 的 是 ，final 修饰 词 还 可 以 修饰 类 名 。 如 果 在 定义 某 个 类 
时 使 用 了 final 修饰 词 ， 那 么 就 不 能 创建 基于 该 类 的 子 类 。 

(6) interface 

使 用 interface 修饰 的 类 称 为 接口 类 。 在 接口 类 中 ， 只 需 声 明 方 法 和 属性 ， 不 需要 为 这 
些 方法 和 属性 编码 和 赋值 . 当 我 们 使 用 implements 关键 字 来 创建 一 个 基于 某 接 口 类 的 类 时 ， 
在 新 创建 的 类 中 ， 必 须 定 义 那些 已 经 在 接口 类 中 声明 的 方法 和 属性 。 

比如 应 用 程序 允许 使 用 MySQL、PostgreSQL 和 SQLite 三 种 类 型 的 数据 库 来 存储 数据 。 
为 了 应 用 程序 在 对 接 这 三 类 数据 库 时 使 用 相同 的 方法 和 属性 ， 我 们 可 以 定义 一 个 接口 类 ， 
然后 在 这 个 接口 类 中 声明 需要 的 方法 和 属性 。 接 下 来 ， 就 可 以 使 用 implements 关键 字 创 建 
三 个 类 ， 分 别针 对 不 同类 型 的 数据 库 来 实现 这 些 方法 和 属性 。 

(7) abstract 

使 用 abstract 修饰 的 类 称 为 抽象 类 。 所 谓 抽象 类 ， 顾 名 思 义 ， 就 是 把 若干 类 中 公用 的 
部 分 抽 离 出 来 ， 放 到 一 个 抽象 类 中 以 减少 脚本 的 元 余 程 度 。 抽 象 类 的 作用 与 接口 类 类 似 。 
不 同 的 是 ， 抽 象 类 中 除了 声明 方法 和 属性 之 外 ， 还 可 以 为 其 添加 功能 或 赋值 。 另 外 ， 需 要 
注意 的 是 ， 在 使 用 抽象 类 时 ， 使 用 的 关键 字 是 extends， 而 不 是 implements。 

在 这 些 类 修饰 词 中 ， 前 四 种 只 适用 于 在 类 中 修饰 方法 名 和 属性 名 ， 第 五 种 则 同时 适用 
于 修饰 类 名 和 类 中 的 方法 名 与 属性 名 ， 而 最 后 两 种 则 只 适用 于 修饰 类 名 。 


12.2.4 ”一些 魔术 方法 


在 第 8 章 学 习 了 两 个 魔术 方法 ”_construct0 和 _ destruct0。 知 道 了 魔术 方法 的 名 称 前 
会 有 两 道 下 夯 线 (“__”) 。 本 小 节 将 了 解 到 另外 三 个 魔术 方法 get0、_ set0 和 __issetO 。 
在 编写 notes 类 的 时 候 ， 大 家 写 了 八 个 类 用 set 和 get 开头 的 方法 用 来 设置 和 获取 存储 
日 志 的 文本 文件 名 称 、 日 志 的 作者 、 写 作 日 期 和 当时 的 天 气 情况 。 在 如 此 短小 精 悍 的 小 程 
序 里 ,就 要 写 八 个 setter 和 getter 方法 。 那 要 是 应 用 程序 的 功能 更 复杂 , 要 写 的 setter 和 getter 
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可 就 太 多 了 。 为 了 避免 出 现 这 种 繁琐 的 情况 ， 可 以 使 用 __set0 和 __getO 魔 术 方 法 。 

使 用 __setO 魔 术 方 法 设置 了 许多 专属 于 某 个 对 象 的 临时 属性 之 后 ， 我 们 可 能 一 时 没有 
办 法 弄 清 楚 是 不 是 设置 了 某 个 属性 。 这 时 ， 就 要 通过 _isset( 方 法 来 进行 判断 ， 以 免 出 现 
使 用 没有 设置 的 属性 的 情况 。 

下 面 来 看 一 个 简单 的 使 用 __set0、__sget0 和 _，_issetO 魔 术 方 法 的 例子 。 

【 例 12.8】 使 用 ”set 站、 _get0 和 _isset0 魔 术 方 法 的 例子 (Wclass.character.php) 。 

<?php 

class character { 


// 设 置 一 个 私有 数组 变量 


private $properties = array(); 


// 定 义 一 个 “get () 魔术 方法 
function get ($property) { 
return Sthis -> properties[$property]; 


// 定 义 一 个 _set () 魔术 方法 
function set ($property, $value){ 

$this -> properties[$property] = $value; 
} 


// 定 义 一 个 isset () 魔术 方法 


function isset ($property) { 


return isset(Sthis -> properties[$property]); 
， 
2 
在 这 个 名 为 character 的 类 中 ， 0 -个 私有 属性 ， 其 数据 类 型 为 数组 ， 然 后 使 
用 et、 _get 和 _isset 定 义 了 三 个 公有 方法 。 其 中 ，__set 方 法 有 两 个 参数 ， 没 有 返回 值 ; 
__get 和 _isset 方法 均 有 个 力 六 前 者 返回 获取 到 的 值 ， 后 者 返回 指定 参数 是 否 存在 的 
判断 结果 ; __unset 方法 有 一 个 参数 ， 且 没有 返回 值 。 
在 定义 好 这 个 类 之 后 ， 我 们 再 实例 化 这 个 类 ， 来 见证 一 下 魔法 的 神奇 。 
【 例 12.9】 创建 一 个 基于 character 的 类 并 使 用 魔术 方法 。 
<?php 
include once('class/class.character.php'); 


// 创 建 一 个 基于 character 类 的 对 象 ， 将 其 命名 为 Schara[11] 


Schara[1] = new character(); 


// 现 在 我 们 定义 一 下 $charal 的 一 些 基本 信息 


Schara[1] -> firstName = 'John'; 

Schara[1] -> lastName = 'Dow'; 

$chara[l] -> gender = 'Male'7 

$chara[l] -> occupation = "Product Manager'; 
Schara[1] -> company = "Parachute Inc.'; 


// 再 创建 一 个 基于 character 类 的 对 象 ， 将 其 命令 为 Schara[2] 


$chara[2] = new character(); 


// 现 在 我 们 定义 一 下 $chara2 的 一 些 基本 信息 


$chara[2] -> firstName = 'Jane'; 
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$chara[2] -> lastName = 'Dow'; 

$chara[2] -> gender = 'Female'; 

$chara[2] -> occupation = "PR Coordinator'; 

$chara[2] -> company = "Parachute Inc.'; 

$chara[2] -> undergrad = "Cornel1l1 University (2007-2011)"'; 
$chara[l2] -> grad = "Stanford University (2011-2013)"'; 


// 再 创建 第 三 个 基于 character 类 的 对 象 ， 将 其 命名 为 chara[3] 


$chara[3] = new character (); 


// 现 在 我 们 定义 一 下 $chara3 的 一 些 基 本 信息 


$chara[3] -> firstName = 'Max'; 
$chara[3] -> lastName = "Poll'; 
$chara[3] -> gender = 'Male'; 
$chara[3] -> occupation = "Sales Director'; 
$chara[3] -> company = "Parachute Inc.'; 
$chara[3] -> undergrad = "Carnegie Mellon University (2003-2007)'; 
> 
<!DOCTYPE html> 
<html> 
<head> 
<title>Name Box</title> 
<meta charset="UTF-8"> 
<link href="css/Ex12-09.css" rel="stylesheet"> 
</head> 
<body> 
<div class="container"> 
<?php 
// 接 着 我 们 输出 一 下 这 三 个 人 的 信息 
foreach ($chara as S$key => $value) { 
echo "<dqiv class = "float-box"><p class="num">#' .$key.' 
</p><p class="name">'. 
($value -> isset('firstName')?$value -> firstName : N/A').' '. 
($value -> isset('lastName') ? S$value -> lastName : 
'N/A').'gnbsp;<span class="title">'. 
($value -> isset('gender') ? 
((Svalue -> gender == 'Male') ? 'Mr.' : 'Mrs./Miss') 
'N/A') .'</span></p><p class="pc">'. 
($value -> ___isset('occupation') ? $value -> occupation 
"N/AA") "</p><p Class="pDC" > 
($value -> isset('company') ? $value -> company : 
'N/A') .'</p><ul class="edu"><]1i>'. 
($value -> isset('undergrad') ? $value -> undergrad : 
NAA SEE 
($value -> isset('grad') ? $value -> grad : 'N/A').' 
</1li></div>'; 
} 
?2> 
</div> 
</body> 
</html> 


例 12.9 创建 了 三 个 基于 character 类 的 实例 ， 分 别 为 charal 、chara2 和 chara3。 然 后 我 
们 访问 并 设置 了 这 三 个 对 象 中 并 不 存在 的 私有 变量 ， 如 firstName、lastName 和 gender 等 。 
接着 在 页 面 的 HTML 部 分 ， 输 出 了 刚才 设置 的 这 些 私 有 变量 。 

神奇 的 事情 出 现 了 : 页 面 没有 报错 ， 并 且 输 出 了 对 应 的 值 。 

在 实例 化 这 个 类 的 页 面 上 ， 我 们 在 输出 某 个 属性 之 前 使 用 ”isset0 判 断 该 属性 是 否 存 
在 。 若 存在 ， 则 输出 该 属性 的 值 ， 若 不 存在 ， 则 输出 “N/A”。 就 像 下 面 这 人 句 脚本 一 样 : 
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($value -> isset('firstName') ? $value -> firstName : 'N/A') 


页 面 中 的 所 有 元 素 都 是 通过 这 种 方式 输出 到 页 面 中 的 。 
这 个 页 面 最 终 的 效果 如 图 12-4 所 示 。 


© DD www.greatwalltea/chapter12/Ex12-09.php 
回 Suggested Sites [~ Web Slice Galley DD Imported Fom 正 


#2 


Jane Dow mrs./Miss 


Parachute Inc. 


e Comel Unversty (2007-2011) 。 Camegle Melon Unversty 
» Stanford Unversty (2011- 


(2003-2007) 
2013) » NA 


图 12-4 魔术 方法 的 使 用 
看 到 这 里 ， 大 家 应 该 明白 了 如 何 使 用 这 些 魔 术 方 法 了 。 不 过 需要 再 提醒 的 是 ， 合 理 使 
用 魔术 方法 才 是 正道 。 虽 然 魔术 方法 可 以 方便 为 对 象 设置 临时 属性 ， 但 是 过 多 地 依赖 魔术 
方法 可 能 会 导致 参数 过 于 复杂 ， 从 而 降低 脚本 的 可 读 性 和 可 维护 性 。 
所 以 ， 在 使 用 这 些 魔术 方法 时 ， 万 望 三 思 。 


12.3 进 阶 OOP 


在 上 一 节 里 ， 我 们 掌握 了 基于 对 象 编程 的 基础 知识 。 在 本 节 里 ， 我 们 将 基于 对 象 编程 
的 一 些 高 级 功能 进行 讲解 。 


12.3.1 ” 摸 清 类 的 情况 

在 有 些 时 候 ， 我 们 需要 对 类 的 情况 进行 摸底 ， 以 便 更 好 地 使 用 类 及 类 中 定义 的 方法 和 
属性 。 因 此 ，PHP 为 我 们 提供 了 一 些 函 数 来 获取 指定 类 的 信息 ， 从 而 摸 清 类 的 情况 。 

如 果实 例 化 一 个 不 存在 的 类 ， 系 统 会 报错 。 为 了 避免 出 现 这 种 情况 ， 我 们 可 以 使 用 
class_exists0 函 数 来 判断 一 个 类 是 否 存 在 。 这 样 一 来 , 我 们 就 可 以 保证 要 实例 化 的 类 是 一 定 


ss 
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存在 的 了 。 
【 例 12.10】 使 用 class_exists0 函 数 判 断 类 是 否 存 在 。 
<?php 


include once('class/class.calc.php'); 


if(class exists('calc')){ 
$calc = new calc(); 
echo $calc -> getVarA().'+"'.$calc ->getVarB().'="'.$calc ->add(); 
} else { 
die (' 不 存在 名 为 "calc" 的 类 。"'); 
2 ! 
在 例 12.10 中 ， 使 用 class_exists0 判 断 类 calc 是 否 存 在 。 若 存在 ， 则 创建 一 个 基于 该 
类 的 名 为 calc 的 对 象 ， 然 后 通过 对 象 调用 getVarA、getVarB 和 add 方法 输出 算式 “5 +6= 
11”; 若 calc 类 不 存在 ， 则 输出 “不 存在 名 为 calc 的 类 ”。 
如 果 想 看 到 类 不 存在 时 脚本 的 运行 结果 ， 可 以 把 include_once 一 句 给 注释 掉 。 
有 的 时 候 ， 我 们 想 知道 现在 可 供 使 用 的 类 有 哪些 ， 以 便 在 其 中 选择 一 些 类 。PHP 提供 
了 一 个 名 为 get_declared_classes 方法 ， 可 以 帮助 我 们 做 到 这 一 点 。 在 例 12.11 的 例 程 中 ， 
我 们 导入 了 类 文件 class.calcphp， 然 后 使 用 var_dump 方法 输出 了 get_declared_classes() 函 
数 的 运行 结果 。 你 会 发 现 get_declared_classes0 函 数 返 回 的 是 一 个 数组 ,数组 中 全 是 字符 串 
类 型 的 元 素 。 每 个 字符 串 元 素 都 代表 着 一 个 已 经 被 声明 的 类 。 滚 动 鼠标 到 页 面 的 最 底 端 ， 
可 以 找到 导入 的 calc 类 。 
【 例 12.11】 使 用 get declared_classes 方法 获取 当前 已 声明 的 所 有 类 。 
<?php 


include once('class/class.calc.php'); 


echo '<pre>';var dump (get declared classes());echo '</pre>'; 

2> 

有 时 候 ， 我 们 在 一 个 页 面 中 会 导入 多 个 类 文件 ， 可 能 还 会 创建 多 个 基于 同一 个 类 的 对 
象 。 创 建 的 对 象 一 多 ， 就 会 让 人 犯 迷 糊 。PHP 提供 了 一 个 名 为 is_a0 的 函数 ， 可 以 帮助 我 
们 判断 某 个 对 象 到 底 是 不 是 基于 某 个 类 的 。 这 个 函数 的 参数 是 某 个 对 象 的 名 称 和 某 个 类 的 
名 称 。 

【 例 12.12】 使 用 is_a0 函 数 判断 某 对象 是 否 为 某 类 的 实例 。 

<?php 


include once('class/class.calc.php'); 


/ /创建 一 个 基于 calc 类 的 实例 


$calc = new calc(); 


// 判 断 对 象 Scalc 是 否 是 一 个 基于 calc 类 的 实例 
echo (is a($calc,'calc'))? "对 象 Scalc 是 一 个 基于 calc 类 的 实例 。' : ' 对 象 $calc 
不 是 一 个 基于 calc 类 的 实例 。' : 
2 
使 用 is_a0 函 数 只 能 判断 某 个 对 象 是 不 是 某 个 已 知 类 的 实例 。 如 果 想 要 知道 某 个 对 象 
到 底 是 哪个 类 的 实例 ， 它 就 无 能 为 力 了 。 不 过 PHP 又 准备 了 一 个 名 为 get_class0 的 函数 ， 
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可 以 帮助 我 们 做 到 这 一 点 。 这 个 函数 只 有 一 个 参数 ， 返 回 一 个 字符 串 类 型 的 值 。 
【 例 12.13】 使 用 get_class0 函 数 查 找 某 对 象 所 属 类 。 


<?php 


> 


class parentClass { 


// 定 义 一 个 公有 属性 用 于 获取 基于 该 类 的 对 象 所 属 的 类 
function getClass() { 
echo get class(); 
echo "<br>'7 
echo get class($this); 
} 
} 


// 创 建 一 个 继承 parentClass 类 的 名 为 childclass 的 类 


class childClass extends parentClass { 
} 
// 创 建 一 个 基于 chilqclass 类 的 名 为 $cc 的 对 象 


$cc = new childClass(); 


// 直 接 调用 get_class 函数 输出 对 象 $cc 所 属 的 类 
echo get class($cc); 
echo '<br>'; 


// 调 用 对 象 Scc 的 公有 方法 getClass 
$cc -> getClass () 


例 12.13 定义 了 两 个 类 ， 分 别 为 parentClass 和 childClass， 其 中 childClass 继承 自 


parentClass。 在 parentClass 中 ， 我 们 定义 了 


出 get_class0 和 get_class($this) 的 值 。 


接着 我 们 创建 了 


所 属 的 类 。 最 后 调用 了 对 象 Scc 的 getClass 方法 。 
这 段 脚 本 运行 的 结果 如 下 : 


childClass 
parentClass 
childClass 


一 行 输 出 的 “childClass” 是 在 对 象 $cc 被 创建 后 直接 输出 的 get_class($cc) 的 值 。 第 


二 行 和 第 三 行 输 


其 中 ， 


第 


里 ， 所 以 返 


-个 公有 方法 getClass， 并 在 当中 使 用 echo 输 


-个 基于 childClass 类 的 对 象 , 名 为 $cc, 然后 使 用 echo 输出 了 对 象 $cc 


H 的 是 通过 $cc 对 象 调用 的 继承 自 parentClass 的 getClass 方法 的 运行 结果 。 


第 二 行为 getClass 方法 中 的 “echo get_class0” 输 出 的 结果 ， 而 第 三 行 则 是 getClass 
方法 中 的 “echo get class($this) ”输出 的 结果 。 
通过 这 段 例 程 ， 我 们 可 以 知道 ， 如 果 使 用 不 带 参数 的 get classO 函 数 ， 返 


回 的 是 


其 


运行 范畴 决定 的 。 在 上 面 这 段 例 程 中 ， 不 带 参数 的 get_class0 就 运行 在 parentClass 的 范畴 
有 回 的 值 为 “parentClass”。 如 果 使 用 带 参数 的 get_class($this)， 那 么 ， 当 我 们 实 


例 化 parentClass 类 时 , $this 指 代 的 就 是 基于 parentClass 类 的 实例 ; 当 我 们 实例 化 childClass 


类 时 ，S$this 


parentClass 


指 代 的 就 是 基于 childClass 类 的 实例 。 在 上 面 这 段 例 程 中 ，childClass 是 继承 自 
条子 类 ， 而 对 象 Sce 又 是 childClass 的 实例 化 对 象 , 所 以 这 时 S$this 指 代 的 应 该 是 
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childClass， 那 么 它 返回 “childClass” 就 无 可 厚 非 了 。 
12.3.2 ”迭代 器 


在 第 6 章 中 ， 我 们 学 习 过 如 何 遍 历数 组 。 我 们 也 可 以 用 相同 的 方法 来 人 毅 历 对 象 中 的 公 
有 属性 和 方法 ， 而 这 种 方法 是 PHP 4 时 代 就 有 的 方法 。 在 PHP 5 时 代 ，PHP 为 我 们 准备 了 
友 代 器 接口 类 ， 通 过 使 用 迭代 器 来 遍历 数组 和 对 象 。 

所 谓 迭 代 器 ， 就 是 用 来 遍历 数组 和 对 象 的 类 。 它 们 通常 使 用 implements 关键 字 定 义 ， 

拥有 统一 的 方法 。PHP 中 提供 的 迭代 器 有 如 下 两 种 。 

口 Iterator: Iterator 称 为 简单 迭代 器 。 它 是 一 个 预定 义 的 接口 类 , 定义 了 current、 key、 
next、rewind 和 valid 五 种 必 选 方法 接口 。 通 过 自 定义 这 些 方法 接口 的 实现 方式 ， 
因地制宜 地 实现 对 数组 和 对 象 的 遍历 。 

口 IteratorAggregate: IteratorAggregate 称 为 聚合 迭代 器 。 它 只 有 一 个 名 为 getIterator 
的 预定 义 的 方法 接口 。 通 过 在 该 方法 接口 中 引用 外 部 迭代 器 ， 可 以 实现 对 数组 和 
对 象 的 遍历 。 

下 面 ， 我 们 就 来 创建 一 个 简单 迭代 器 和 一 个 聚合 迭代 器 。 前 者 用 来 读 取 日 志文 件 目录 

下 的 所 有 日 志文 件 中 的 日 志 记 录 ; 后 者 则 用 来 读 取 日 志文 件 目录 下 的 所 有 日 志文 件 。 

【 例 12.14】 简单 迭代 器 notesIterator (\\class.notesInterator) 。 

<?php 


// 定 义 一 个 用 于 遍历 使 用 notes 类 创建 的 所 有 日 志 的 夫 代 器 
class notesIterator implements Iterator { 

// 定 义 一 些 私有 变量 

Private $files = array(); 

Private $notes = array(); 

Private $position; 


// 定 义 一 些 常量 
const FILE ROOT = 'file/'; 


// 初 始 化 迭代 器 
function _ construct(){ 
// 获 取 指定 目录 下 的 所 有 日 志文 件 
$dh = opendir (self: :FILE ROOT) 
$i = 0; 
while ($fn = readdir($dh)) { 
if(substr($fn, -6) == '.notes' && 
Strlen{(s$stEn) 2S 2Y 寺 
Sthis => files[$i] = $fn; 
a 


} 

// 获 取 所 有 日 志文 件 中 存储 的 日 志 

Si=0; 

foreach (Sthis -> files as $fnid => $fn) { 


$fh = fopen(self::FILE ROOT.$fn, 'r'); 
while($line = fgets ($fh)){ 
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if($line <> "----Start----N\rNn")T 
Sthis -> notes[$i] = mb substr($line,0,mb strlen 
($line, ‘'utf-8')-2,'utf-8°'); 


Si 
} 
} 
fclose ($fh); 
上 
// 初 始 化 迭 化 器 指针 


Sthis -> position = 0; 


} 


function rewind(){ 
Sthis -> position = 0; 
} 


function current(){ 
return Sthis -> notes[$this -> position]; 


} 


function next(){ 
++$this -> position; 
} 


function key(){ 
return Sthis -> position; 


} 


function valid(){ 
return isset($this -> notes[$this -> position]); 


} 


?> 


在 notesIterator 迭代 器 中 ， 我 们 先 封装 了 三 个 私有 变量 fles、notes 和 position。 其 中 ， 
files 和 notes 的 数据 类 型 为 数组 ， 而 position 数据 类 型 未 定 。 接 着 我 们 封装 了 一 个 常量 
FILE_ ROOT， 其 值 为 “file/”。 表 明 存 储 日 志文 件 的 目录 名 称 。 

在 _constructO 魔 术 方法 中 , 我们 通过 opendir()、readdir()、fopen() 和 fgets() 函 数 将 “file/” 
目录 下 的 日 志文 件 和 这 些 日 志文 件 里 的 日 志 记录 读 取 到 了 files 和 notes 两 个 私有 属性 中 。 

随后 ， 我 们 定义 了 iterator 接口 类 预定 义 的 五 个 方法 接口 rewind、current、next、key 
和 valid 的 实现 方式 ， 这 个 迭代 器 就 写 好 了 。 

现在 再 来 看 看 如 何 定 义 聚合 迭代 器 。 通 过 引入 外 部 迭代 器 ， 我 们 定义 迭代 器 的 过 程 会 
变 得 更 加 简单 。 

【 例 12.15-1】 聚合 迭代 器 filesIterator (\\class.fileIterator.php) 。 


<?php 
class filesIterator implements IteratorAggregate { 


// 定 义 一 些 私 有 变量 


private $files = array()s 


// 定 义 一 些 常量 


“1s 
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const FILE ROOT = "file/ "7 


function construct() { 


// 获 取 指 定 目录 下 的 所 有 日 志文 件 

$dh = opendir (self::FILE ROOT); 

$= 0 

while ($fn = readdir($dh)) { 

1F(SubDste ($fn, =6) == "notes” ee 
strlen($fn) > 2) { 

Sthis -> files[$i] = $fn; 
Sitt? 


} 


function getIterator(){ 
return new ArrayIterator ($this->files); 
} 
} 


2> 

在 filelterator 类 中 ， 我 们 先 封装 了 一 个 私有 变量 $file， 其 数据 类 型 为 数组 。 接 着 封装 
了 一 个 常量 FILE_ ROOT, 其 值 为 “file/”, 表 明 存储 日 志文 件 的 目录 。 然 后 ,我 们 使 用 opendirO) 
和 readdir() 函 数 将 指定 目录 中 的 所 有 后 级 名 为 “.notes” 的 日 志文 件 的 文件 名 存储 到 私有 属 
性 $files 数组 中 。 

最 后 ,我 们 实现 了 IteratorAggregate 类 预定 义 的 getIterator 方法 接口 ， 引 入 了 一 个 外 部 
迭代 器 ArrayIterator 用 来 遍历 filesInterator 类 的 私有 属性 files。 

现在 来 创建 基于 notesIterator 类 和 filesIterator 类 的 实例 ， 将 其 命名 为 Sni 和 $fi。 

【 例 12.15-2】 创建 一 个 基于 notesIterator 类 的 名 为 $ni 的 实例 (\Ex12-15.php)。 

<?php 


include once('class/class.notesiterator.php'); 
include once('class/class.filesiterator.php'); 


// 使 用 notesIterator 和 迭代 器 遍历 所 有 日 志文 件 中 的 日 志 
$ni = new notesIterator () 


foreach ($ni as S$key => $value) { 
echo '<pre>';var dump ($key, $value);echo '</pre>'; 
} 


// 使 用 filesIterator 迭代 器 遍历 存储 日 志文 件 的 目录 中 所 有 日 志文 件 
$fi = new filesIterator(); 


foreach ($fi as $key => $value) { 
echo '<pre>';var dump ($key,$value);echo ‘</pre>'; 
} 
2 
在 上 面 这 段 脚本 中 ， 我 们 实例 化 了 notesIterator 类 和 filesnotes 类 ， 并 将 实例 化 后 的 对 
象 分 别 赋值 给 变量 $ni 和 $f。 然 后 使 用 foreach 语句 输出 了 这 两 个 对 象 的 内 容 
例 12.15-2 中 的 脚本 运行 结果 如 图 12-5 所 示 。 
通过 图 12-5 可 以 发 现 Sni 和 $fi 对 象 分 别 把 私有 属性 Snotes 和 $files 里 的 内 容 全 部 读 取 
出 来 了 ， 我 们 不 需要 定义 任何 getter 方法 就 可 以 获取 这 两 个 私有 属性 中 的 所 有 内 容 。 是 不 
是 很 方便 呢 ? 
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DD) www.greatwalltea/chapte x Nd 


© Dwwwgreatwalltea/chapter12/Ex12-15.php 
深 应 用 回 suggested Sites 门 Web Slice Gallery [Imported From 正 


int(@) 
咱 string(48)“ 白 秋明 | 睛 |2813 年 16 月 97 日 | 中 秋 节 快乐 !1" 


int(1) 

string(48)“ 白 秋明 | 睛 |2813 年 16 月 97 日 | 中 秋 节 快乐 !" 

int(2) 

string(158) "张晓明 | 上 晴 |2813 年 16 月 97 日 | 今天 天 空 放晴 。 北 方 秋 日 的 早晨 青 京 干 列 。 沐浴 在 阳光 下 ， 吹 着 冷风 ， 心 情 却 莫名 地 好 了 起 来 。” 
int(3) 

string(95) “张晓明 | 晴 12613 年 19 月 e8 日 | 今天 是 节 后 上 班 的 第 一 天 ， 工 作 不 怎么 在 状态 。” 

int(@) 

string(18) "55m956eL5pi0.notes” 


int(1) 
string(18) "5byg5pmT5piO.notes” 


图 12-5 notesIterator 实例 化 后 的 输出 结果 


12.3.3 ”数组 对 象 


在 PHP 4 时 代 , 数组 和 对 象 是 不 同 的 。 但 是 到 了 PHP 5 时 代 , 除了 有 数组 和 对 象 之 外 ， 
还 有 数组 对 象 。 通 过 数组 对 象 ， 我 们 可 以 很 方便 地 使 用 基于 对 象 的 方法 访问 和 修改 数组 。 
下 面 来 看 看 这 个 奇妙 的 数组 对 象 。 

在 PHP5 中 ，PHP 预定 义 了 一 个 ArrayObject 类 。 这 个 类 有 如 下 几 个 常用 的 方法 。 

口 append: 该 方法 是 用 来 向 数组 添加 元 素 的 。 

口 getIterator: 该 方法 会 创建 一 个 简单 迭代 器 对 象 ， 并 返回 该 对 象 。 这 样 我 们 就 可 以 
使 用 Iterator 对 象 预定 义 的 方法 来 遍历 数组 。 
offsetExists: 该 方法 用 来 判断 某 个 指定 的 索引 是 否 已 经 定义 。 
offsetGet: 该 方法 用 来 获取 某 个 指定 的 索引 对 应 的 值 。 
offsetSet: 该 方法 用 来 设置 某 个 指定 的 索引 对 应 的 值 。 
offsetUnset: 该 方法 用 来 取消 对 某 个 指定 索引 的 定义 。 
现在 我 们 来 定义 一 个 数组 对 象 。 
【 例 12.16】 创建 数组 对 象 (\Ex12-16.php〉。 


<?php 
// 定 义 一 个 数组 
$users = new ArrayObject (array ("hasin"=>"hasin@pageflakes.com", 
"afif"=>"mayflower@phpxperts.net", 
"ayesha"=>"florence@pageflakes.net")); 


// 定 义 一 个 迭代 器 


Siterator = Susers->getIterator () 


// 使 用 和 欠 代 器 响 历数 组 
while ($iterator->valid()) 


echo "{$iterator->key()}'s Email address is 
{$iterator->current () }<br>"; 
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Siterator->next () 7 


} 
BS 
在 上 面 这 段 脚本 中 ， 首 先 创建 了 一 个 基于 ArrayObject 类 的 对 象 ， 将 其 命令 为 Suser， 
用 来 存储 用 户 相关 信息 。 接 着 使 用 $user 对 象 的 getIterator 方法 创建 了 一 个 迭代 器 ， 最 后 遍 
历 这 个 迭代 器 中 的 所 有 信息 。 


12.3.4 对象 序 列 化 


对 象 序列 化 是 指 把 已 创建 的 基于 某 个 类 的 对 象 转换 成 可 在 页 面 间 传递 的 字 节 流 ， 这 也 
是 对 象 序列 化 诞生 的 最 重要 的 一 个 原因 。 它 包括 两 个 步骤 : 序列 化 和 反 序 列 化 。 前 者 通过 
serialize() 函 数 序列 化 一 个 对 象 ， 使 其 转化 成 一 个 二 进 制 字符 串 ， 后 者 则 使 用 unserializeO 
函数 将 这 个 二 进 制 字符 串 转换 成 之 前 被 序列 化 的 对 象 。 

通过 这 样 的 方法 在 页 面 间 传 递 的 对 象 ， 其 数据 和 内 存 结构 都 是 完整 的 。 不 过 需要 注意 
的 是 ， 序 列 化 函数 serialize 只 能 序列 化 对 象 的 属性 ， 不 能 序列 化 对 象 的 方法 ， 也 就 是 说 在 
需要 反 序 列 化 由 某 个 对 象 转化 而 来 的 二 进 制 字符 串 时 ， 需 要 导入 派生 该 对 象 的 类 。 

我 们 看 一 个 对 象 序列 化 的 例子 。 

【 例 12.17】 对 象 的 序列 化 〈\NExl2-17.php) 。 

<?php 

// 开 启 session 


session start(); 


// 导 入 calc 类 所 在 的 类 文件 


include once('class/class.calc.php'); 


// 创 建 一 个 基于 calc 类 的 对 象 ， 将 其 命名 为 $c 


$c = new calc(); 


// 序 列 化 该 实例 ， 并 将 序列 化 后 的 字 节 流 存 入 名 为 $cstring 的 session 变量 中 
$ SESSION['cString'] = serialize($c); 


/ /创建 一 个 链接 ， 用 来 转向 另 一 个 页 面 
echo '<a href="Ex12-18.php"> 开 始 传 递 对 象 </a>'; 
2> 
在 例 12.17 中 开启 了 session 功能 ， 打 算 使 用 session 变量 在 两 个 页 面 间 传 递 由 calc 类 
派生 出 的 对 象 。 接着 我 们 导入 了 calc 类 , 创建 了 一 个 基于 calc 类 的 对 象 ， 并 将 其 命名 为 $c。 
然后 使 用 serializeO 函 数 将 对 象 $c 序列 化 ， 并 将 序列 化 后 的 二 进 制 字符 串 存 入 session 变量 
$_SESSION['cString] 中 。 最 后 ， 在 页 面 上 输出 了 一 个 链接 ， 链 向 对 象 传递 的 目的 页 面 。 
下 面 ， 我 们 来 编写 对 象 传递 的 目的 页 面 。 
【 例 12.18】 对 象 的 反 序 列 化 (\Ex12-18.php)〉。 
<?php 
// 开 启 session 


session start(); 


// 导 入 派生 对 象 $c 的 类 


include once('class/class.calc.php'); 
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// 判 断 session 变量 cString 是 否 已 设置 ， 若 已 经 设置 则 反 序 列 化 该 sesssion 变量 
if(isset($ SESSION['"cString"])){ 
$c = unserialize($ SESSION['cString']); 


// 使 用 对 象 $c 的 getVarRA() 、getVarB () 和 add 方法 输出 "5 + 6 = 11" 
echo $c -> getVarA().' + '.$c -> getVarB().' = '.$c -> add(); 
} else { 


// 若 不 存在 名 为 cString 的 session 变量 ， 则 输出 如 下 内 容 
echo “对 象 未 定义 。 ' 
2> l 
在 这 个 页 面 中 ， 我 们 也 开启 了 session 功能 以 便 接 受 由 其 他 传递 过 来 的 session 变量 。 
由 于 serialize0 函 数 只 能 传递 对 象 的 属性 ， 而 不 能 传递 方法 ， 所 以 我 们 在 对 象 传递 的 目的 页 
面 上 也 导入 了 派生 对 象 $c 的 calc 类 。 接 着 判断 session 变量 是 否 已 设置 若 未 设置 ， 则 输 
出 “对 象 未 设置 ”; 若 已 设置 ， 则 使 用 unserialize0 函 数 反 序列 化 存储 在 session 变量 中 的 
二 进 制 字 节 流 ， 并 将 反 序 列 化 后 的 对 象 存 储 在 同名 对 象 gc 中 。 最 后 ， 我 们 使 用 对 象 $c 的 
getVarA、getVarB 和 add 方法 输出 了 “5 +6=11” 这 个 算式 。 
通过 上 面 两 个 例 程 , 我 们 知道 了 如 何 通过 序列 化 对 象 , 从 而 方便 地 在 页 面 间 传 递 对 象 。 
这 里 由 于 篇 幅 的 原因 ， 使 用 了 最 简便 的 session 变量 。 试 问 ， 如 果 不 使 用 session 变量 ， 如 
何 实现 类 似 例 12.17 和 例 12.18 之 间 的 对 象 传递 呢 ? 大 家 可 以 思考 一 下 。 


12.3.5 “对象 的 克隆 


为 了 讲 清楚 对 象 的 克隆 ， 我 们 先 来 看 一 段 脚本 。 
【 例 12.19】 对 象 的 克隆 (WEx12-19.php)。 


<?php 
include once('class/class.calc.php'); 


$samplel = new calc(); 


// 使 用 对 象 Ssamplel 的 setter 方法 设置 私有 属性 a,b 的 值 分 别 为 10,20 
$samplel->setVarA(10); 

$samplel->setVarB (20); 

// 输 出 对 象 Ssamplel 的 add 方法 的 执行 结果 

echo $samplel->getVarA().' + '.$samplel->getVarB().' = '.$samplel->add(). 
<br 


// 将 对 象 Ssamplel 赋值 给 对 象 $sample2 
$sample2 = $samplel; 


// 使 用 对 象 Ssample2 的 setter 方法 设置 私有 属性 a,b 的 值 分 别 为 8, 9 


$sample2->setVarA(8); 
$sample2->setVarB(9); 


// 输 出 对 象 Ssamplel 的 add 方法 的 执行 结果 
echo $samplel->getVarA().' +".$samplel->getVarB() .' = '.$samplel->add(). 
A 


// 将 对 象 $sample2 克隆 给 对 象 Ssample3 
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$sample3 = clone $samplel; 


// 使 用 对 象 $sample3 的 setter 方法 设置 私有 属性 a,b 的 值 分 别 为 10，20 
$sample3->setVarA(10); 
$sample3->setVarB (20); 


// 再 次 输出 对 象 Ssamplel 的 adq 方法 的 执行 结果 
echo $samplel->getVarA().' + '.$samplel->getVarB().' = '.$samplel->add(); 
2> 

在 上 面 的 脚本 中 ， 首 先 创建 了 一 个 基于 calc 类 的 对 象 ， 将 其 命名 为 samplel。 接 着 使 
用 samplel 对 象 的 setter 方法 将 其 私有 属性 a 和 的 值 分 别 设置 为 10 和 20; 将 对 象 samplel 
赋值 给 变量 sample2。 然 后 使 用 sample2 的 setter 方法 将 其 私有 属性 a 和 b 的 值 分 别 设置 为 
8 和 9。 最 后 把 对 象 sample2 克隆 给 sample3， 并 通过 其 setter 方法 将 其 私有 属性 a 和 ob 分 
别 设置 为 10 和 20。 

在 使 用 setVarA 和 setVarB 方法 修改 了 samplel 对 象 的 私有 属性 后 ， 我 们 使 用 对 象 
samplel 的 getVarA、getVarB 和 add 方法 输出 了 一 个 算式 。 在 把 samplel 对 象 赋值 给 sample2 
后 ， 使 用 对 象 samplel 的 getVarA、getVarB 和 add 方法 又 输出 了 一 个 算式 。 在 克隆 后 ,我 
们 又 一 次 使 用 了 对 象 samplel 的 相同 方法 输出 了 一 个 算式 。 

大 家 可 以 猜 一 下 ， 这 三 次 输出 的 结果 是 一 样 的 吗 ? 结果 应 该 是 “10+20=30”， 还 是 
“8+9=17” 呢 ? 

首先 ， 第 一 次 输出 的 结果 为 “10 + 20 = 30”， 因 为 我 们 使 用 两 个 setter 方法 修改 私有 
属性 a 和 的 值 , 第 二 次 输出 的 结果 为 “8+9= 17”, 因为 在 使 用 sample2 对 象 的 两 个 setter 
方法 修改 了 其 私有 属性 的 同时 ，samplel 对 象 的 私有 属性 也 被 修改 了 。 而 第 三 次 输出 的 结 
果 依 然 为 “8 + 9= 17”， 因 为 sample3 对 象 和 samplel 对 象 是 两 个 相互 独立 、 互 不 影响 的 
两 个 对 象 , 使 用 sample3 对 象 的 setter 方法 修改 其 私有 属性 与 samplel 的 私有 属性 没有 什么 
关系 。 

如 果 大 家 还 想 不 通 的 话 ， 可 以 把 对 象 Ssamplel 和 $sample2 之 间 的 关系 看 成 是 一 个 人 在 
照 镜子 ， 这 个 人 做 的 任何 动作 ， 镜 子 里 的 他 都 会 重复 得 一 模 一 样 ， 这 就 是 “赋值 ”的 内 涵 。 
而 在 对 象 Ssamplel 和 $sample3 之 间 存 在 的 却 不 是 这 种 关系 ,它们 更 像 是 一 对 双胞胎 : 有 着 
惊人 相似 的 外 貌 ， 却 有 着 绝对 独立 的 属性 ， 这 才 是 “克隆 ”的 内 涵 。 


12.3.6 方法 链 


方法 链 是 PHP 5 中 引入 的 一 种 让 我 们 可 以 在 对 象 被 返回 的 同时 使 用 该 对 象 的 方法 和 属 
性 ， 就 像 下 面 这 样 : 


$someObject -> getObjectOne () -> getObjectTwo() -> callMethodofObjectTwo(); 


在 上 面 这 个 示例 中 ， 有 一 个 名 为 $someObject 的 对 象 ， 用 getObjectOne 的 方法 来 获取 
名 为 objectOne 的 对 象 。 而 objectOne 对 象 用 getObjectTwo 方法 来 获取 名 为 objectTwo 的 对 
象 。 而 objectTwo 的 对 象 有 一 个 名 为 callMethodOfObjectTwo 方法 用 来 执行 最 后 的 步骤 。 

在 这 种 情况 下 ， 我 们 可 以 使 用 上 述 链 式 引 用 的 方法 来 执行 objectTwo 对 象 的 
callMethodOfObjectTwo 方法 。 

现在 我 们 来 看 一 个 实际 的 例子 。 
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【 例 12.20】 定义 名 为 emailBuilder 的 类 (\\class.emailbuilder.php) 。 


<?php 


// 定 义 名 为 methodchain 的 类 


class emailBuilder { 
/7 定义 一 些 私 有 变量 
private $from; 
private Sto7 
private $subject; 
private $body; 
private $cc; 
private $date; 


// 初 始 化 emailBuilder 的 时 间 变 量 


function construct(){ 


$this -> date = 


} 
// 设 置 发 件 人 


date('Y 年 m 月 d 日 '); 


function from ($from){ 


$this -> from = 


return S$this; 


} 
// 设 置 收 件 人 


function to (Sto) { 


$from; 


Sthis -> to = $to; 


return S$this; 


| 
// 设 置 邮件 主题 


function subject ($subject) { 


$this -> subject 


return S$this; 


} 
// 设 置 邮件 内 容 


= $subject; 


function body ($body) { 


Sthis -> body = 


return S$this; 


} 
// 设 置 邮件 抄 送 对 象 


function ce (Sce) { 


$body; 


Sthis => ce = Kees 


return S$this; 


} 
// 输 出 邮件 


function buildEmail() { 
echo '<b>>> 发 件 人 : </b>'.$this -> from.'<br>'. 
"<b>>>> 收 件 人 : </b>' .Sthis -> to.'<br>'. 


"<b>>>> 抄 送 : 
"<b>>>> 日 期 : 
"<b>>>> 主 题 : 
"<b>>>> 内 容 : 


< 

</b>". Sthis -> date. <br>”. 

</b>' -Sthis -> subject.'<br>'. 

< /b> <DE><D>>>><XD>SLEhas > oodY > 


天 贡生 过 
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} 

2 

在 这 个 名 为 emailBuilder 的 类 中 ， 我 们 定义 了 一 些 私 有 变量 做 为 邮件 的 组 件 。 然 后 再 
定义 了 一 些 公有 方法 ， 用 来 修改 这 些 组 件 内 容 。 最 后 定义 了 一 个 名 为 buildEmail 的 公有 方 
法 ， 用 来 输出 整 封 邮件 的 内 容 。 

在 这 里 ， 需 要 提醒 注意 的 是 ， 所 有 用 来 修改 邮件 组 件 的 方法 返回 的 值 都 是 Sthis， 而 这 
才 是 使 用 方法 链 的 关键 。 

【 例 12.21】 使 用 方法 链 定义 并 输出 邮件 内 容 (\Ex12-21.php)〉。 


<?php 
include once('class/class.emailbuilder.php'); 


// 定 义 了 一 个 名 为 $eb 的 emailBuilder 对 象 


Seb = new emailBuilder(); 


// 使 用 方法 链 定 义 并 输出 邮件 内 容 

$eb -> from('sender@example.com') 
-> to('receiver@example.com') 
-> cc('ccto@example.com') 
-> subject (' 这 是 一 封 用 emailBuild 类 创建 的 邮件 ') 
-> body (' 要 使 用 emailBuilder， 先 创建 一 个 基于 emailBuilder 类 的 对 象 ， 
然后 使 用 方法 链 定义 并 输出 邮件 内 容 。"' ) 
-> buildEmail (); 

be 


在 上 面 这 段 例 程 中 ， 我 们 定义 了 一 个 基于 emailBuilder 类 的 对 象 Seb， 然 后 使 用 方法 链 
设置 并 输出 了 邮件 的 内 容 ， 如 图 12-6 所 示 。 


] www.greatwalltea/chapt x 和 


~- 3 © 口 wwwgreatwalltea/chapter12/Exl2-21.php 
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>> 发 件 人 : sender@example.com 

>>> 收 件 人 : receiver@example.com 

>>> 抄 送 : ccto@example.com 

>>> 日 期 : 2013 年 10 月 13 日 

2 这 是 一 封 用 emailBuild 类 创建 的 邮件 

>>> 内 容 : 

>>> 要 使 用 emailBuilder， 先 创建 一 个 基于 emailBuilder 类 的 对 象 ， 然 后 使 用 方法 链 定义 并 输出 邮件 内 容 。 


图 12-6 ”使 用 方法 链 设置 并 输出 邮件 内 容 


12.4 设计 模式 


基于 对 和 象 的 编程 ， 其 目的 是 为 了 简化 编程 过 程 、 减 少 开发 时 间 和 简洁 脚本 代码 。 如 果 
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使 用 得 当 ， 基 于 对 象 的 编程 可 以 极 大 地 提升 脚本 的 性 能 。 早 在 1972 年 ，Eric Gamma 和 他 
的 三 个 朋友 就 针对 基于 对 象 的 编程 提出 了 一 系列 的 基于 对 象 编程 的 标准 化 模式 ， 这 些 标 准 
化 模式 被 称 为 “设计 模式 ”， 而 这 四 个 人 则 被 人 形象 地 称 为 “四 人 帮 (Gang of Four) ”。 
看 到 这 个 极度 抽象 的 名 词 ， 很 多 同学 都 会 不 由 自主 地 产生 旦 难 的 心理 。 因 为 我 当初 接 
触 这 个 “设计 模式 ”的 时 候 ， 也 有 相同 的 想法 。 不 过 ， 大 家 再 想 一 想 ， 如 果 设计 模式 是 为 
了 减缓 编码 效率 和 脚本 执行 效率 ， 肯 定 没有 人 用 。 所 以 ， 它 的 首要 目的 肯定 是 提升 编码 效 
率 和 脚本 执行 效率 。 为 了 实现 这 个 目的 ，“ 四 人 帮 ” 在 他 们 常年 地 编码 实践 中 总 结 并 抽象 
出 了 一 些 最 佳 实践 ， 形 成 了 一 些 标准 化 的 编码 模式 。 
其 实 ， 大 家 在 编码 的 过 程 中 ， 肯 定 也 用 到 了 一 些 与 这 些 最 佳 实践 类 似 的 提升 编码 效率 
和 减少 编码 量 的 方法 。 只 是 不 知道 这 些 方法 可 以 被 抽象 到 “模式 ”这 么 一 个 高 度 。 
在 本 节 里 ， 我 们 就 来 简单 地 了 解 一 下 在 PHP 编码 过 程 中 常常 会 用 到 的 一 些 设计 模式 。 


12.4.1 策略 模式 (Strategy) 


策略 模式 通常 用 在 某 个 算法 需要 有 多 个 不 同 的 变 体 以 应 对 不 同情 况 的 时 候 。 打 个 比 
方 ， 如 果 你 需要 定义 一 个 类 用 来 创建 图 片 。 但 是 图 片 有 很 多 的 格式 ， 在 某 些 情况 下 ， 想 使 
用 该 方法 创建 JPEG 格式 的 图 片 ; 而 在 另 一 些 情况 下 , 想 使 用 该 方法 创建 GIF 格式 的 图 片 。 
则 可 以 把 这 个 类 定义 成 下 面 这 个 样子 : 

<?php 


class createImage { 
private $images; 


function createJPEG() { 
$this -> images = 'image.jpg'; // 此 处 代码 只 用 于 示例 ， 不 能 创建 JPEG 
格式 的 图 片 
return Sthis -> images; 


} 


function createGIF() { 
Sthis -> images = 'image.gif'; // 此 处 代码 只 用 于 示例 ， 不 能 创建 GIF 
格式 的 图 片 
return S$this -> images; 
} 
} 
2% 
在 这 个 类 中 ,可 以 为 每 一 种 需要 的 图 片 格式 定义 一 个 creator 方法 。 但 是 一 旦 需要 的 图 
片 格式 比较 多 ， 这 个 类 就 会 被 撑 到 很 大 。 到 时 候 ， 不 光 是 脚本 难以 维护 ， 脚 本 的 性 能 也 会 
大 打折 扣 。 为 了 解决 这 个 问题 ， 我 们 可 以 先 定义 一 个 抽象 类 (abstract) ， 然 后 再 定义 基于 
这 个 抽象 类 的 针对 不 同 格式 的 子 类 来 具体 地 创建 指定 格式 的 图 片 。 这 样 一 来 ， 类 的 数量 虽 
然 增加 了 ， 但 是 不 仅 维护 起 来 要 容易 的 多 ， 而 且 脚本 性 能 也 会 得 到 提升 。 如 果 需 要 新 的 图 
片 格式 ， 再 创建 一 个 基于 抽象 类 的 新 类 就 好 了 。 
按照 这 样 的 想法 ， 上 面 这 段 脚本 就 可 以 修改 成 下 面 这 个 样子 : 


<?php 
abstract class CreateImage { 
Protected $image; 
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function createImage (){ 
上 
} 


class createJPEG extends createImage { 


function createImage(){ 
$this -> image = 'image.jpeg'; // 此 处 代码 只 用 于 示例 ， 不 能 创建 JPEG 
格式 的 图 片 


return Sthis -> image; 


} 


class createGIF extends createImage { 


function createImage(){ 
Sthis -> image = 'image.gif'; // 此 处 代码 只 用 于 示例 ， 不 能 创建 GIF 
格式 的 图 片 


return Sthis -> image; 


} 
2 


12.4.2 工厂 模式 (Factory) 


在 策略 模式 一 节 里 ， 我 们 定义 了 一 个 抽象 类 ， 然 后 从 这 个 抽象 类 中 派生 出 了 两 个 具体 
类 。 如 果 我 们 直接 使 用 这 两 个 具体 类 来 创建 对 象 ， 然 后 使 用 该 对 象 的 creator 方法 创建 图 片 
的 话 ， 那 这 个 抽象 类 的 存在 就 没有 意义 了 。 

通常 情况 下 ， 我 们 在 使 用 策略 模式 定义 了 一 些 策略 后 ， 会 使 用 工厂 模式 来 执行 这 些 策 
略 。 换 名 话说， 策略 模式 就 像 是 工厂 的 高 层 制订 的 生产 策略 ， 而 工厂 模式 是 车 间 主 任 根据 
在 生产 过 程 中 遇 到 的 不 同情 况 在 这 些 策 略 中 选择 具体 的 策略 ， 以 保证 生产 的 进行 。 所 以 工 
厂 模式 通常 是 和 策略 模式 一 起 使 用 的 。 

按照 这 个 思路 ， 我 们 可 以 在 上 面 这 段 脚 本 末尾 加 上 一 个 名 为 imageFactory 的 类 ， 来 具 
体 执行 创建 图 片 的 过 程 。 这 个 imageFactory 类 的 脚本 如 下 : 


class imageFactory { 
private static $format; 


static function creator ($format) { 
if (!isset(self::$format)) { 
self::$format = $format; 


} 


Switch (self::$format) { 

case "JPEG' : 
return new createJPEG () 
break; 

case 'GIF': 
return new createGIF (); 

default: 
return “无 法 创建 图 片 。 没 有 找到 指定 的 图 片 格式 对 应 的 类 。 ' ; 
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在 这 个 imageFactory 类 中 , 我 们 定义 了 一 个 静态 私有 属性 $format 用 来 存放 所 需 图 片 的 
格式 。 接 着 ,定义 了 一 个 静态 的 名 为 creator 的 方法 ， 该 方法 会 根据 私有 属性 的 值 来 判断 需 
要 创建 的 对 象 。 如 果 所 需 图 片 的 格式 为 PEG， 在 使 用 imageFactory 类 的 creator 方法 时 ， 
指定 其 format 参数 为 “JPEG”， 那 么 creator 方法 返回 的 就 是 一 个 createJPEG 实例 ， 若 指 
定 其 format 参数 为 “GIF”， 那 么 creator 方法 返回 的 就 是 一 个 createGIF 实例 。 由 于 这 两 
个 实例 拥有 相同 的 方法 和 属性 ， 我 们 可 以 将 creator 方法 返回 的 对 象 存储 到 一 个 变量 中 ， 然 
后 通过 这 个 变量 来 使 用 createJPEG 和 createGIF 类 的 方法 ， 就 像 下 面 这 样 : 


$format = 'JPEG'; 


$creator = imageFactory::creator ($format); 


if(!is string($creator)){ 

echo $creator -> createImage () 7 
} else { 

echo S$creator; 


} 
12.4.3 单 体 模式 (Singleton) 


单 体 模式 通常 用 在 某 个 类 只 需要 被 实例 化 一 次 不 允许 被 再 次 实例 化 的 情况 下 。 比 如 ， 
我 们 在 定义 上 面 这 个 imageFactory 类 的 时 候 ， 就 使 用 了 单 体 模式 。 在 讲解 什么 是 单 体 模式 
之 前 ， 先 把 12.4.2 小 节 末 尾 的 那 段 脚本 修改 成 如 下 : 

$format = "JPG' 7 

$creator = imageFactory::creator ($format) ; 

$format = 'JPEG'; 


$creator = imageFactory::creator ($format); 


if(!is string(Screator) ){ 

echo S$creator -> createImage () 7 
} else { 

echo $creator; 

} 

在 这 段 脚 本 中 ， 我 们 定义 了 一 个 名 为 $format 的 变量 ， 并 为 其 赋值 “JPG”， 然 后 将 这 
个 变量 代入 imageFactory::creator($format) 中 ， 想 要 创建 一 个 用 来 创建 JPG 格式 图 片 的 
creator 对 象 。 我 们 知道 , 之 前 并 没有 定义 用 来 创建 卫 G 格式 图 片 的 creator 类 ， 所 以 这 里 应 
该 输出 的 是 如 下 所 示 报 错 信息 : 

无 法 创建 图 片 。 没 有 找到 指定 的 图 片 格式 对 应 的 类 。 


接着 , 将 变量 $format 的 值 修改 为 "JPEG”, 然后 再 次 使 用 imageFactory::creator($format) 
实例 化 imageFactory 类 ， 以 创建 一 个 用 来 创建 JPEG 格式 图 片 的 creator 对 象 。 我 们 知道 ， 
之 前 已 经 定义 了 用 来 创建 JPEG 格式 图 片 的 creator 类 ， 所 以 这 里 应 该 可 以 返回 一 个 基于 
createJPEG 类 的 对 象 。 

可 出 乎 我 们 意料 的 是 ， 这 有 段 脚本 执行 的 结果 依然 是 上 面 那 条 报错 信息 。 

为 什么 呢 ? 
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因为 我 们 在 imageFactory 类 中 定义 其 属性 和 方法 时 ， 使 用 了 static 修饰 词 ， 这 就 使 得 
imageFactory 类 中 的 属性 和 方法 在 imageFactory 类 第 一 次 被 实例 化 后 就 固定 了 下 来 ， 不 会 
再 发 生变 化 了 。 换 名 话说， 通过 使 用 static 修饰 词 实现 了 单 体 模式 ， 或 者 说 ， 使 用 单 体 模 
式 定 义 一 个 类 ， 就 是 通过 为 该 类 创建 静态 属性 和 方法 ， 来 实现 该 类 只 会 被 实例 化 一 次 的 目的 。 


12.4.4 ”观察员 模式 (Observer) 


在 PHP 应 用 程序 中 ， 通 过 对 数据 的 操控 来 实现 许多 功能 。 在 很 多 情况 下 ， 某 个 数据 的 
改变 会 影响 到 该 程序 的 很 多 方面 。 以 购物 网 站 上 的 商品 列表 页 面 为 例 ， 当 一 个 位 于 中 国 的 
消费 者 打开 该 页 面 时 ， 商 品 的 价格 是 以 人 民 币 计价 的 ， 而 当 一 个 位 于 美国 的 消费 者 打开 该 
页 面 时 ， 商 品 的 价格 则 是 以 美元 计价 的 。 在 这 两 次 访问 中 ， 只 有 一 个 参数 不 一 样 ， 那 就 是 
用 户 发 起 访问 的 地 点 。 也 就 是 说 ， 用 户 发 起 访问 的 地 点 这 个 参数 影响 到 了 商品 列表 页 面 上 
显示 的 商品 价格 。 一 旦 用 户 发 起 访问 的 地 点 发 生 了 变化 ， 商 品 列表 上 展示 的 商品 的 价格 也 
会 发 生变 化 。 

我 们 可 以 把 商品 列表 上 展示 的 每 件 商 品 当 成 是 一 个 观察 员 对 象 。 它 们 作为 观察 员 的 职 
责 是 盯 住 用 户 发 起 访问 的 地 点 这 个 参数 ， 一 旦 这 个 参数 发 生 了 变化 ， 立 即 更 新 其 价格 属性 
并 输出 。 

为 了 实现 这 个 功能 ， 我 们 可 以 使 用 观察 员 模式 。 先 创建 一 个 名 为 accessLocObserver 的 
接口 类 ， 在 其 中 定义 一 个 setPrice 方法 ， 就 像 下 面 这 样 。 

【 例 12.22】 创建 一 个 观察 员 接口 类 〈\Ninterface.accesslocobserverphp) 。 

< 

全 accessLocObserve 的 接口 类 ， 用 来 观察 accessLocation 类 的 变化 


interface accessLocObserver { 


// 一 旦 accessLocation 类 中 发 生 了 变化 ， 就 设置 物品 的 价格 属性 


function setPrice($o0bj); 


} 

人 六 

在 这 个 接口 类 中 ， 定 义 了 一 个 setPrice 方法 ， 该 方法 有 一 个 参数 ， 就 是 由 被 观察 的 类 
派生 出 来 的 对 象 。 

接着 ， 创 建 一 个 名 为 accessLocation 的 类 ， 该 类 拥有 一 个 名 为 $location 的 私有 属性 。 
一 旦 该 属性 被 重新 设置 ， 就 会 触发 notifyObservers 方法 。 另 外 ， 该 类 还 有 一 个 静态 属性 
$instance 和 静态 方法 getInstance， 可 供 我 们 在 实现 accessLocObserver 类 的 具体 类 中 通过 
“::” 实 例 化 accessLocation 类 ， 并 使 用 其 拥有 的 方法 。 除 此 之 外 ，accessLocation 类 还 有 一 
个 名 为 observers 的 私有 属性 ， 用 来 存储 注册 到 该 类 的 所 有 对 象 ， 一 个 名 为 getLocation 的 
方法 ， 用 来 获取 当前 用 户 发 起 访问 的 地 点 ; 一 个 名 为 registerObserver 的 方法 ， 用 来 将 指定 
的 对 象 注册 到 由 accessLocation 类 派生 出 的 对 象 中 ; 一 个 名 为 notifyObservers 的 方法 ， 用 
来 通知 已 经 注册 到 由 accessLocation 类 派生 出 的 对 象 中 的 所 有 对 象 。 

这 个 名 为 accessLocation 的 类 如 下 。 

【 例 12.23】 创建 被 观察 员 观 察 的 类 accessLocation (\\ class.accesslocation.php) 。 


<?php 


// 创 建 一 个 名 为 accessLocation 类 
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class accessLocation { 


// 创 建 一 个 私有 属性 instance， 用 来 在 类 中 实例 化 自身 


static Private $instance = NULL; 


// 创 建 一 个 私有 属性 ， 用 来 存储 用 户 发 起 访问 的 地 点 
private $location = "中 国 大 陆 "; 


// 创 建 一 个 私有 数组 ， 用 来 存储 注册 到 accessLocation 类 的 观察 员 


private $observers = array(); 


// 创 建 一 个 静态 方法 ， 用 来 在 类 中 实例 化 自身 
static function getInstance() { 
if(self::$instance == NULL) { 
self::$instance = new accessLocation(); 


} 


return self::$instance; 


用 
// 创 建 用 来 获取 当前 用 户 发 起 访问 地 点 的 方法 


function getAccessLocation() { 
return Sthis -> location; 


} 
// 创 建 用 来 设置 当前 用 户 发 起 访问 地 点 的 方法 


function setAccessLocation($newLocation) { 
$this -> location = $newLocation; 


// 在 设置 当前 用 户 发 起 访问 地 点 的 同时 ， 通 知 所 以 注册 到 该 类 的 观察 员 
$this -> notifyObservers () : 
} 


// 创 建 用 来 将 观察 员 注 册 到 accessLocation 类 的 方法 
function registerObservers (Sobserver) { 
Sthis -> observers[] = $observer; 


} 
// 创 建 用 来 通知 观察 员 的 方法 


function notifyObservers(){ 
foreach ($this -> observers as S$observer) { 
Sobserver -> setPrice($this); 


b 


> 


在 上 面 的 脚本 中 ， 值 得 我 们 注意 的 是 ， 在 setAccessLocation 方法 中 除了 设置 私有 属性 
$location 的 值 之 外 ， 还 触发 了 notifyObservers 方法 。 

接 下 来 ， 还 要 定义 一 个 用 来 实现 accessLocObserver 接口 类 的 方法 ， 并 将 其 命名 为 
productItem 。 

【 例 12.24】 创建 productItem 类 (Wclass.productitem.php) 。 


<?php 
include once('interface/interface.accesslocobserver.php'); 
include once('class/class.accesslocation.php'); 


// 创 建 用 来 实现 accessLocObserver 接口 类 的 商品 类 


class productItem implements accessLocObserver { 


"Ds 
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private S$price; 

private $forexRates = array (' 美 国 ' => 0.1634, 
' 欧 洲 ' => 0.1206, 
"英国 ' => 0.1024， 
"中 国 香港 ' => 1.2669， 
"中 国 台 湾 ' => 4.7992， 
"澳大利亚 ' => 0.1726， 
站 本 "=> 16 1074 
"韩国 ' => 175.0426); 


// 初 始 化 productItem 类 并 将 其 注册 到 accessLocation 类 中 
function _construct($price){ 
Sthis -> price = $price; 
accessLocation: :getInstance () ->registerObservers ($this); 


// 定 义 accessLocObserver 接口 类 的 setPrice 方法 接口 的 实现 
function setPrice (Sobj){ 
if($ob]j instanceof accessLocation) { 
Sthis -> price = Sthis -> price * 
$this -> forexRates[accessLocation::getInstance () -> 
getAccessLocation()]; 


ly 
// 定 义 获取 产品 当地 价格 的 方法 


function getPrice(){ 
switch (accessLocation: :getInstance () -> getAccessLocation()) { 

case "美国 ' : 
SSiGmE SR 
break; 

case "欧洲 ' : 
Saliqgn = we ws 
break; 

case "英国 ' : 
Saign = Rm 
break; 

case ' 中 国 香港 ' : 
sign = "$m 
break; 

case ' 中 国 台湾 ': 
$sign = “NES 2 
break; 

case ' 澳 大 利 亚 ': 
$sign = "$ ™; 
break; 

case "日 本 ": 
Yaign = Wy 
break; 

case "韩国 ' : 
Ssign = "WH "> 
break; 

default: 
ssign = "¥"> 
break; 
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return accessLocation::getInstance () -> getAccessLocation(). 
" 售 价 : ' .$sign.number format (floatval ($this ->price),2); 


} 

2> 

在 这 个 名 为 productItem 的 类 中 ， 我 们 为 其 定义 了 两 个 私有 属性 ， 分 别 为 $price 和 
$forexRates 。 在 初始 化 这 个 类 时 ， 设 置 了 私有 属性 $price 的 值 ， 并 将 该 类 注册 到 了 由 
accessLocation 类 派生 出 来 的 对 象 中 。 然 后 ， 在 productItem 类 中 分 别 定 义 了 一 个 setPrice 
和 一 个 getPrice 方法 ， 前 者 是 我 们 在 accessLocObserver 接口 类 中 规定 的 必须 实现 的 方法 ， 
用 来 根据 accessLocation 对 象 的 私有 属性 $location 的 值 的 不 同 而 设置 不 同 的 价格 ; 而 
getPrice 是 我 们 自行 定义 一 个 用 来 获取 当前 价格 的 方法 。 

最 后 ， 我 们 要 用 accessLocation 和 productItem 两 个 类 来 实现 本 节 开 始 的 时 候 描述 的 功 
能 , 即 一 旦 用 户 发 起 访问 的 地 点 发 生 了 变化 , 商品 列表 上 展示 的 商品 的 价格 也 会 发 生变 化 。 

【 例 12.25】 商品 售 价 随 用 户 访问 地 点 的 不 同 而 发 生变 化 (\Ex12-25.php)。 

<?php 


include once('class/class.accesslocation.php'); 
include once('class/class.productitem.php'); 


$pi = new productItem(1588); 


全 天 
<!DOCTYPE html> 
<html> 
<head> 
<title> 随 地 点 变化 的 商品 售 价 </title> 
</head> 
<body> 
<form action="?location" method="post"> 
访问 地 点 
<select name="location"> 
<option value=" 美 国 "> 美国 </option> 
<option value=" 欧 洲 "> 欧洲 </option> 
<option value=" 英 国 "> 英国 </option> 
<option value=" 澳 大 利 亚 "> 澳大利亚 </option> 
<option value=" 日 本 "> 日 本 </option> 
<option value=" 韩 国 "> 韩国 </option> 
<option value=" 中 国 香港 "> 中 国 香港 </option> 
<option value=" 中 国 台湾 "> 中 国 台 湾 </option> 
</select> 
<input type="submit"” value=" 显 示 售 价 "> 
</form> 
hr /> 
<?php 
// 输 出 商品 的 初始 价格 


echo $pi -> getPrice()."<br>"; 


// 判 断 用 户 是 否 提交 了 表单 
if(isset($ REQUEST['location']))t{ 


// 若 用 户 提交 了 表单 ， 则 根据 用 户 的 选择 输出 对 应 的 结果 
$location = $ REQUEST['location']; 
Switch ($location) { 

case "美国 了 


accessLocation: :getInstance () -> setAccessLocation(' 美 国 '); 
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echo $pi -> getPrice()."<br>"; 
break; 

case "欧洲 ' : 
accessLocation: :getInstance () -> setAccessLocation(' 欧 洲 '); 
echo $pi -> getPrice()."<br>"; 
break; 

case ' 英 国 ': 
accessLocation: :getInstance() -> setAccessLocation( ' 英 国 '); 
echo S$pi -> getPrice()."<br>"; 
break; 

case ' 澳 大 利 亚 ': 
accessLocation: :getInstance ()-> setAccessLocation( ' 澳 大 利 亚 ') ; 
echo S$pi -> getPrice()."<br>"; 
break; 

case ' 日 本 ': 
accessLocation: :getInstance() -> setAccessLocation(' 日 本 '); 
echo $pi -> getPrice()."<br>"; 
break; 

case ' 韩 国 ': 
accessLocation: :getInstance() -> setAccessLocation(' 韩 国 '); 
echo $pi -> getPrice()."<br>"; 
break; 

case ' 中 国 香港 ' : 
accessLocation: :getInstance () -> setAccessLocation(' 中 国 香港 '); 
echo $pi -> getPrice()."<br>"; 
break; 

case ' 中 国 台湾 ': 
accessLocation: :getInstance () -> setAccessLocation(' 中 国 台 湾 ') ; 
echo S$pi -> getPrice()."<br>"; 
break; 


2 : 

</body> 

</html> 

在 这 个 页 面 上 ， 从 productItem 类 中 派生 出 了 一 个 名 为 $pi 的 对 象 ， 并 在 页 面 上 输出 了 
S$pi 对 象 的 初始 价格 。 然 后 根据 用 户 的 选择 输出 对 应 地 区 的 商品 价格 。 

在 脚本 中 ， 值 得 我 们 注意 的 是 ， 通 过 accessLocation::getInstance 方法 来 使 用 
accessLocation 实例 化 后 的 静态 方法 。 由 于 getInstance 返回 的 是 accessLocation 的 一 个 实例 ， 
因此 ， 我 们 可 以 直接 在 其 后 面 链 式 使 用 setAccessLocation 方法 ， 从 而 根据 用 户 的 选择 设置 
当前 accessLocation 对 象 的 私有 属性 $location 的 值 。 


中 后 雹 点 变化 的 商品 售 价 。 x 
CC DD www.greatwalltea/chapter12/Ex12-25.php 
深 应 用 回 Suggested Stes 门 Web Slice Gallery ©) Imported From IE 


访问 地 点 [美男 国 江东 车 兢 


中 国 大 陆 售 价 : ¥1,588.00 


图 12-7 页 面 输出 的 原始 价格 
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现在 我 们 直接 单 击 图 12-7 页 面 上 的 “显示 售 价 ” 按 钮 ， 页 面 上 显示 的 内 容 如 图 12-8 
所 示 。 


€ 了 © Dwwwogreatwalltea/chapter12/Ex12-25.php?location 
洪 应 用 回 Suggested Sites 门 Web Sice Gallery DD Imported Fom 正 


访问 地 点 | 美国 。 国 | 昌 示 篇 价 ] 


中 国 大 陆 售 价 : ¥1,588.00 


| 美国 售 价 : $ 259.48 


图 12-8 页 面 输出 的 美国 当地 价格 


12.5 习 题 


(1) 请 使 用 策略 模式 编写 一 个 抽象 类 createFile。 然 后 创建 两 个 派生 自 createFile 类 的 
子 类 ， 分 别 为 createTXT 和 createCSV， 用 于 创建 文本 文件 和 CSV 文件 。 
(2) 请 使 用 工厂 模式 编写 一 个 抽象 类 fileFactory， 用 于 创建 文本 文件 和 CSV 文件 。 


"Ys 
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在 上 一 章 里 , 我 们 学 习 了 如 何 使 用 基于 对 象 编程 的 方法 来 缩写 PHP 脚本 , 也 了 解 了 如 
何 使 用 一 些 常用 的 设计 模式 来 简化 我 们 日 常 的 编码 工作 。 如 果 我 们 能 够 从 一 个 更 为 宏观 的 
视角 来 看 PHP 应 用 程序 的 开发 过 程 ， 应 该 从 应 用 程序 的 架构 谈 起 。 在 本 章 , 会 了 解 一 种 非 
常 流行 的 应 用 程序 架构 ， 它 被 称 为 MVC， 是 Model-View-Controller 的 简称 。 
在 PHP 快速 应 用 开发 中 ，MVC 扮演 了 十 分 重要 的 角色 。 现 如 今 ， 市 场 上 有 着 形 形 色 
色 的 基于 MVC 架构 的 开发 框架 。 在 它们 当中 ， 使 用 比较 广泛 的 开发 框架 有 CakePHP、 
CodeIgniter 和 Zend Framework 等 等 。 它们 虽然 不 全 是 基于 对 象 编程 的 框架 , 但 是 却 都 是 基 
于 MVC 架构 的 应 用 框架 。 使 用 这 些 框架 可 以 极 大 地 加 快 应 用 程序 的 开发 。 

在 这 样 的 背景 下 ， 白 手 起 家 写 应 用 程序 的 日 子 似乎 已 经 到 头 了 ， 因 为 更 多 的 应 用 程序 
是 基于 开发 框架 的 。 如 果 想 要 更 好 地 了 解 PHP 应 用 程序 的 开发 过 程 ， 就 必须 了 解 一 下 什么 
是 MVC。 


13.1 MYVC 大 起 底 


在 本 小 节 ， 我 们 来 具体 了 解 一 下 MVC 架构 是 什么 。 
13.1.1 什么 是 MVC 


前 面 说 到 ，MVC 是 Model-View-Controller 的 简称 。 顾 名 而 思 义 ,所谓 的 MVC 架构 有 
着 三 个 重要 的 组 成 部 分 , 它们 分 别 是 模型 (Model) 、 视 图 (View) 和 控制 器 (Controller) 。 
那么 这 三 位 又 都 是 干什么 的 呢 ? 

简单 地 说 ， 模 型 (Model) 是 数据 库 和 第 三 方 服务 互动 的 对 象 ， 它 定义 了 应 用 程序 的 
业务 逻辑 。 所 谓 业 务 逻 辑 ， 指 的 是 任何 与 通过 存储 数据 和 使 用 第 三 方 服务 来 达到 业务 需求 
相关 的 东西 。 如 果 应 用 程序 需要 访问 数据 库 ， 那 么 为 应 用 程序 添加 功能 的 代码 就 应 该 被 写 
入 模型 对 象 中 。 

视图 (View) 则 是 所 有 与 信息 呈现 有 关 的 元 素 的 集合 ,通常 情况 下 ， 视 图 文件 是 一 些 
HTML、CSS 和 JavaScript 文件 。 换 名 话说， 所 有 用 户 看 到 的 和 与 之 互动 的 页 面 都 是 视图 
页 面 。 一 个 视图 页 面 只 有 在 控制 器 与 模型 互动 产生 了 用 户 需 要 的 结果 后 才 会 被 调用 ， 并 通 
过 视图 将 结果 返回 给 用 户 。 

控制 器 (Controller) 则 是 用 来 联系 用 户 、 模 型 和 视图 的 中 间 部 件 ， 它 隔离 并 沟通 了 业 
务 逻 辑 和 结果 呈现 。 通 常情 况 下 ， 控 制 器 是 用 户 访问 的 起 始点 ， 因 为 所 有 的 用 户 请 求 都 由 
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控制 器 处 理 。 当 控制 器 接收 到 用 户 请 求 时 ， 会 根据 用 户 请 求 调用 模型 方法 ， 通 过 模型 获取 
用 户 所 需 的 数据 , 然后 调用 视图 文件 , 将 用 户 所 需 的 数据 通过 视图 文件 返回 给 用 户 。 图 13-1 
展示 了 模型 、 视 图 和 控制 器 的 互动 过 程 。 

在 图 13-1 中 ， 用户 向 控制 器 发 出 请 求 ， 控制 嚣 根 
据 用 户 请 求 调用 相应 的 模型 ， 并 将 模型 返回 的 用 户 所 
需 结果 通过 视图 返回 给 用 户 。 

需要 注意 的 是 ， 该 图 只 是 一 个 原理 示意 图 。 在 实 
际 的 应 用 程序 开发 中 ， 并 不 是 所 有 的 用 户 请 求 都 需要 
调用 模型 或 视图 的 。 有 的 用 户 请 求 涉及 到 的 模型 可 能 
非常 复杂 ， 一 个 模型 解决 不 了 问题 ， 这 时 就 需要 调用 
多 个 模型 ， 而 其 中 肯定 会 有 些 模型 做 为 中 间 件 而 不 需 
要 调用 视图 。 而 有 的 时 候 ， 则 完全 不 需要 框架 ， 直 接 
由 控制 器 将 用 户 需求 转向 视图 ， 继 而 将 内 容 呈 现 给 
用 户 。 


图 13-1 模型 、 视 图 和 控制 器 


13.1.2 为 什么 要 使 用 MVC 


其 实 通过 图 13-1， 大 家 应 该 可 以 猜 到 为 什么 我 们 要 使 用 MVC 架构 。 如 果 你 还 没有 猜 
透 ， 那 再 看 一 遍 MVC 架构 的 工作 流程 : 当 控制 器 接收 到 用 户 发 起 的 访问 请 求 后 ， 会 根据 
用 户 的 请 求 通过 模型 获取 用 户 请 求 的 数据 ; 在 取得 用 户 请 求 的 数据 后 ， 将 数据 通过 视图 展示 
给 用 户 。 控 制 器 、 模 型 和 视图 就 像 一 个 团队 中 紧密 配合 的 成 员 ， 各 司 其 职 ， 实 现 共同 的 目 
标 。 在 这 个 过 程 ， 业 务 罗 辑 、 数 据 读 写 和 内 容 呈 现 被 分 隔 成 了 三 个 独立 而 又 相互 联系 的 部 件 。 

使 用 MVC 模型 ， 我 们 可 以 并 行 开 发 应 用 的 业务 逻辑 、 数 据 读 写 和 内 容 呈 现 部 件 ， 加 
快 开发 进程 。 同 时 ， 由 于 这 三 个 部 分 相互 独立 ， 我 们 对 其 中 某 个 部 件 的 修改 完全 不 会 对 其 
他 两 个 部 件 造成 影响 。 由 于 这 两 个 显而易见 的 好 处 ， 使 用 MVC 模型 进行 PHP 应 用 程序 的 
开发 成 了 我 们 的 不 二 选择 。 


13.1.3 ”常用 的 MVC 框架 


为 了 进一步 加 快 应 用 程序 的 开发 ， 市 场 上 已 经 有 很 多 成 熟 的 开发 框架 供 我 们 选择 。 在 
这 些 开放 框架 中 ， 有 不 少 还 是 开源 的 框架 。 使 用 这 些 开 发 框架 ， 就 可 以 站 在 巨人 的 肩膀 上 
开始 我 们 的 开发 工作 ， 获 得 事半功倍 的 效果 。 

下 面 就 来 了 解 一 下 这 些 开源 的 MVC 开发 框架 。 

(1) CodeIgniter 

CodeIgniter 框架 是 世界 上 第 一 个 ,也 是 最 简洁 的 一 个 基于 MVC 模型 的 PHP 应 用 程序 
开发 框架 。 它 是 由 EllisLab 开发 和 维护 。 虽 然 EllisLab 将 CodeIgniter 定义 为 一 个 开源 的 
MVC 开发 框架 ， 不 过 其 对 CodeIgnite 的 使 用 还 是 有 着 严格 的 控制 。 

CodeIgniter 有 着 完善 的 开发 文档 和 高 质量 的 代码 ， 甚 至 于 有 些 开发 框架 ， 比 如 
KohanaPHP， 深 受 CodeIgniter 框架 的 影响 。 虽 然 CodeIgniter 在 流行 程度 上 可 能 比 不 上 下 
面 介绍 的 一 些 开发 框架 ， 这 完全 是 由 于 ElliseLab 对 CodeIgniter 的 管控 策略 造成 的 ， 不 过 
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EllisLab 正在 对 这 一 问题 做 出 改变 。 

(2) Zend Framework 

Zend Framework 是 一 套 可 扩展 、 轻 耦合 的 脚本 库 。 这 套 基 于 MVC 架构 的 脚本 库 可 以 
做 为 PHP 应 用 程序 的 基础 架构 以 加 快 应 用 程序 的 开发 。 该 框架 由 Zend Technologies 开发 
并 维护 ， 适 用 于 大 型 应 用 程序 的 开发 。 

(3) CakePHP 

CakePHP 是 目前 最 受 欢迎 的 基于 MVC 架构 的 PHP 应 用 程序 开发 框架 。 与 前 两 个 框架 
不 同 的 是 ，CakePHP 的 开发 与 维护 不 是 由 某 一 个 组 织 完成 的 ， 它 有 着 庞大 的 开发 者 社 群 和 
广泛 的 群众 基础 。 

在 CakePHP 开发 框架 中 ,规约 性 是 首要 的 。 只 有 当 控 制 器 、 模 型 和 视图 的 命名 完全 符 
合 规 范 时 ， 它 们 才 会 有 机 的 结合 起 来 。 


13.2 KISSMVC: 一 个 简单 的 MVC 框架 


在 上 一 节 里 ， 我 们 知道 了 什么 是 MVC 架构 和 为 什么 要 使 用 MVC 架构 进行 应 用 程序 
的 开发 。 另 外 ， 我 们 还 了 解 了 当前 市 场 上 可 供 选 择 的 一 些 基 于 MVC 架构 的 PHP 应 用 程序 
开发 框架 。 当 使 用 这 些 开 发 框架 进行 应 用 程序 的 开发 时 ， 除 了 需要 掌握 PHP 脚本 的 相关 知 
识 和 开发 技能 之 外 ， 还 得 了 解 这 些 框 架 的 内 在 机 制 。 有 些 框架 很 简单 、 上 手 也 很 容易 ， 而 
有 些 框架 的 学 习 曲 线 就 很 长 ， 需 要 我 们 投入 大 量 的 时 间 和 精力 。 

为 了 能 更 快 地 了 解 基于 MVC 的 PHP 应 用 程序 开发 ， 在 本 节 里 ， 我 们 将 解剖 一 个 简单 
上 且 开 源 的 基于 MVC 的 开发 框架 KISSMVC。 据 KISSMVC 发 布 网 站 上 的 介绍 ， 
KISSMVC 框架 是 这 个 世界 上 体积 最 小 、 且 最 简单 的 基于 MVC 的 PHP 应 用 程序 开发 框架 。 
其 代码 简单 易 读 ， 几 分 钟 之 内 就 能 够 掌握 其 核心 功能 。 


13.2.1 ”KISSMVC 框架 概述 


KISSMVC 框架 的 核心 文件 只 有 一 个 ， 而 这 个 文件 也 仅 有 16 KB 大 小 。 该 文件 为 我 们 
提供 了 三 个 抽象 类 ， 分 别 如 下 。 
口 KISS_ Model 类 : 该 类 是 一 个 简单 的 基于 对 象 关系 映射 “ORM) 的 数据 库 操 控 类 。 
在 KISSMVC 框架 中 ， 主 要 用 于 操控 存储 在 关系 型 数据 库 中 的 数据 。 
口 KISS_View 类 : 该 类 是 一 个 简单 的 模板 系统 。 在 KISSMVC 框架 中 ， 负 责 数据 的 
呈现 。 
口 KISS_Controller 类 : 该 类 是 KISSMVC 框架 的 中 间 件 。 可 以 解析 用 户 请 求 、 并 根 
据 用 户 请 求 调用 相应 的 基于 KISS Model 和 KiSS_View 类 的 子 类 和 对 象 。 
在 本 小 节 中 ， 为 了 便于 开发 ， 我 们 将 直接 从 KISSMVC 框架 的 发 布 网 站 
Chttp:/Wkissmvc.com) 下 载 名 为 “KISSMVC Blank Project” 的 压缩 包 文 件 。 然 后 将 下 载 下 来 
的 压缩 包 解压 放 到 Apache 服务 器 的 根 目 录 下 的 名 为 chapter13 的 子 目录 中 。 
解压 后 的 目录 结构 如 下 : 
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在 这 个 解压 后 的 目录 中 ， 核 心 文件 为 kissmvc_corephp， 上 文 提 到 的 三 个 抽象 类 就 在 
这 个 文件 中 ， 后 续 我 们 会 对 该 文件 中 的 内 容 进 行 详细 地 解读 。 

在 该 目录 下 ， 我 们 还 有 一 个 名 为 kissmvc php 的 文件 。 在 这 个 文件 中 ， 定 义 了 三 个 派 
生 自 kiss_mvc.php 中 的 三 个 抽象 类 的 普通 类 ， 后 续 在 创建 控制 器 、 模 型 和 视图 对 象 都 是 基 
于 这 三 个 普通 类 的 。 这 样 做 的 好 处 是 可 以 防止 kissmvc_core.php 中 的 抽象 类 被 算 改 ， 同 时 
也 方便 了 我 们 在 普通 类 中 定义 新 的 通用 方法 和 属性 。 

我 们 还 有 一 个 名 为 index.php 的 文件 ， 这 个 文件 是 与 用 户 交 互 的 唯一 入 口 。 基 于 
KISSMVC 框架 的 PHP 应 用 程序 就 是 通过 该 文件 与 用 户 互动 的 。 

除 此 之 外 ， 在 kissmvc 目录 下 ， 还 有 一 个 重要 的 名 为 app 的 子 目 录 。 该 目录 为 应 用 目录 ， 
PHP 应 用 程序 的 所 有 内 容 都 需要 放 在 app 目录 下 相应 的 子 目录 中 。 其 中 子 目 录 的 内 容 如 下 。 

口 Controllers 目录 : 用 来 存放 所 有 的 控制 器 文件 ; 

口 Models 目录 : 用 来 存放 所 有 的 模型 文件 ; 

口 Views 目录 : 用 来 存放 所 有 的 视图 文件 。 

关于 kissmve 目录 下 的 其 余 文 件 和 文件 夹 的 作用 ， 我 们 会 在 用 到 它们 的 时 候 再 进行 简 
单 地 介绍 。 

当 我 们 把 下 载 下 来 的 压缩 包 解 压 好 之 后 ， 将 框架 目录 下 的 index.php 文件 中 的 
WEB_FOLDER 常量 由 之 前 的 “/kissmvc/” 改 成 “/chapter/”。 然 后 打开 浏览 器 , 访问 chapter13 
目录 ,会 看 到 如 图 13-2 所 示 的 内 容 : 


DD) SSMVC - Simple PHP M x 
€ SC 口 wwwgreatwalltea/chapterl3/ 
党 应 用 [Suggested Sites 门 Web Slice Gallery D Imported From IE 


Hello World! 


图 13-2 基于 KISSMVC 框架 开发 PHP 应 用 程序 项 目的 初始 页 面 


a 
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另外 需要 说 明 的 是 ，KISSMVC 框架 支持 通过 分 布 式 配置 文件 (.htaccess) 改造 URL。 
在 第 9 章 里 ， 我 们 学 习 了 如 何 通 过 URL 在 页 面 间 传递 参数 。 以 http://www.example.com/ 
users.php?id=a_florrick 为 例 , id=a_florrick 就 是 我 们 通过 URL 传递 的 参数 。 但 是 这 样 的 URL 
对 于 搜索 引擎 来 说 是 不 太 友 好 的 。 另 外 ， 想 让 人 记 住 这 样 的 URL 也 比较 困难 。 

如 果 我 们 用 Apache 服务 器 做 为 Web 服务 器 并 在 Apache 服务 器 上 开局 mod rewrite 模 
块 ， 那 么 就 可 以 在 分 布 式 配置 文件 〈(.htaccess) 里 添加 相应 的 URL 改写 规则 ， 从 而 实现 通 
过 类 似 于 http://www.example.com/main/users/a_florrick/ 这 样 的 URL 来 访问 这 个 页 面 的 目 
的 。 大 家 可 以 比较 一 下 这 两 个 URL， 哪 一 个 看 上 去 更 容易 理解 呢 ? 


13.2.2 框架 入 口 (index.php) 


在 上 一 节 里 ， 我 们 提 到 框架 根 目录 下 的 index.php 文件 是 用 户 和 框架 交互 的 唯一 入 口 。 
这 就 意味 着 , 用 户 所 有 的 请 求 都 会 通过 index.php 文件 传递 给 相应 的 控制 器 。 那么 , 我 们 就 
先 来 看 看 index.php 文件 里 有 些 什么 内 容 。 

在 index.php 文件 开始 , 开启 了 应 用 程序 调试 功能 并 通过 .htaccess 文件 定义 了 URL 改写 规 
则 。 然 后 ， 通 过 下 面 两 行 脚本 定义 了 KISSMVC 框架 的 两 个 核心 常量 : 一 个 是 应 用 程序 所 
在 路 径 , 而 另 一 个 则 是 框架 所 在 路 径 。 它 们 分 别 对 应 框架 下 的 “app” 目 录 和 框架 所 在 目录 。 


define('APP PATH','app/'); //with trailing slash pls 
define('WEB FOLDER','/chapter13/'); //with trailing slash pls 


接着 通过 自动 化 全 局 变量 * SGLOBALS 定义 了 网 站 的 标题 。 


$GLOBALS['sitename']="'KISSMVC - Simple PHP MVC Framework'; 


然后 使 用 require0 函 数 导 入 了 kissmvc.php 文件 。 在 该 文件 中 , 定义 了 三 个 派生 自 框架 
核心 文件 中 的 三 个 抽象 类 的 普通 类 。 这 三 个 普通 类 分 别 是 Controller、Model 和 View。 


require('kissmvc.php'); 


最 后 定义 了 一 个 基于 Controller 类 的 对 象 : 


$controller = new Controller (APP PATH.' controllers/ 7WEB_FOLDER,，'main' ， 
"index') 


通过 这 四 个 部 分 ， 我 们 构建 了 应 用 程序 的 访问 入 口 。 在 这 四 个 部 分 中 ， 最 为 关键 的 是 


: 关于 分 布 式 配置 文件 的 详细 介绍 ， 可 以 参考 Net Tutsplus 网 站 上 的 一 篇 名 为 “An In Depth Guide to 
Imod rewrite for Apache” 的 文章 (http://net.tutsplus.com/tutorials/other/a-deeper-look-at-mod rewrite-forapache/) 。 
”所 谓 自动 化 全 局 变量 ， 指 的 是 在 脚本 所 有 作用 域 中 都 起 作用 的 变量 。 
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定义 了 一 个 基于 Controller 类 的 名 为 $controller 对 象 。 正 是 通过 该 对 象 ， 我 们 才能 看 到 如 图 
13-2 所 示 的 页 面 。 


13.2.3 ”控制 器 (KISS_Controller) 


我 们 刚刚 讲 到 ， 在 index.php 文件 中 ， 定 义 了 一 个 基于 Controller 类 的 名 为 $controller 
的 对 象 。 通 过 该 对 象 ， 连 通 了 KISSMVC 框架 的 控制 器 、 模 型 和 视图 ， 完 成 了 在 页 面 上 呈 
现 用 户 所 请 求 的 数据 的 任务 。 

在 本 小 节 里 , 我 们 就 来 详细 了 解 一 下 KISSMVC 框架 的 控制 器 类 一 一 KISS_Controller。 

在 KISS_Controller 类 中 ， 我 们 主要 实现 了 如 下 三 个 功能 : 

口 通过 explode_ http request 和 parse_http_request 方法 解析 用 户 请 求 ， 分 离 出 所 需 的 
控制 器 和 行为 的 名 称 ; 

口 通过 route_ request 方法 调用 所 需 的 控制 器 和 行为 ; 

口 通过 request_not found 方法 在 无 法 找到 所 需 控 制 器 和 行为 时 返回 “404 无 法 访问 ” 

页 面 。 

在 下 面 的 脚本 中 ， 定 义 了 六 个 受 保护 的 属性 ， 分 别 用 来 存放 控制 器 文件 路 径 
($controller path) 、 框 架 路 径 〈$web_ folder) 、 用 户 请 求 信息 ($request_uri parts) 、 控 
制 器 名 称 〈$controller) 、 行 为 名 称 〈$action) 和 用 户 通过 URL 传递 的 参数 〈$params) 。 

随后 ， 我 们 使 用 _construct 魔术 方法 初始 化 了 这 些 受 保护 的 属性 并 导入 相应 的 控制 器 
文件 。 


// 控 制 器 类 
// 解 析 用 户 请 求 ， 定 位 相应 控制 器 。 


abstract class KISS Controller { 
protected S$controller_path='../app/controllers/"; //with trailing slash 
protected S$web folder="'/'; //with trailing slash 
protected $request uri parts=array(); 
protected $controller; 
protected $action; 
protected $params=array (); 


function _construct($controller path, $web folder, $default controller, 
$default action) { 

$this->controller path=$controller path; 

$this->web folder=$web folder; 

$this->controller=$default controller; 

$this->action=$default action; 

$this->explode http request()->parse http request()->route request(); 
} 


// 该 方法 用 于 解析 用 户 请 求 ， 并 将 包含 用 户 请 求 信息 的 字符 串 存 入 属性 request_uri parts 中 
function explode http request() { 


} 

// 该 方法 用 于 从 属性 request_uri parts 中 分 离 出 响应 用 户 请 求 的 控制 器 、 行 为 和 用 户 通过 
URL 传递 的 参数 

function Parse http request() { 
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// 该 方法 根据 分 离 出 来 的 控制 器 和 行为 名 称 查找 并 导入 相应 的 控制 器 和 行为 文件 


function route _ request() { 
上 


// 该 方法 定义 了 在 无 法 找到 指定 的 控制 器 时 输出 的 404 页 面 
function request not found($msg="'') { 


} 
lj 


下 面 ， 我 们 来 详细 看 看 KISS_Controller 类 中 定义 的 四 个 方法 。 
(1) 解析 用 户 请 求 (explode_http_request 和 parse_http_request) 
function explode http request() { 
$requri = $ SERVER['REQUEST URI']; 
if (strpos($requri,$this->web folder)===0) 
$requri=substr ($requri, strlen (Sthis->web folder)); 
Sthis->request uri parts = $requri ? explode('/',$requri) : array(); 
return S$this; 


} 


在 explode_http_request 方法 中 , 通过 $_ SERVER 变量 获取 了 用 户 请 求 的 URI 信息 。 由 
于 框架 的 根 目 录 不 一 定 是 服务 器 的 根 目录 , 我 们 可 以 通过 比 对 该 URI 和 框架 根 目录 来 调 校 
该 URI 信息 ， 将 URI 信息 中 的 框架 根 目录 去 掉 。 若 调 校 后 的 URI 信息 长 度 为 0， 则 将 
requst_uri_parts 属性 的 值 定义 为 一 个 空 数组 。 否 则 ， 则 使 用 explode0 函 数 分 离 URI 信息 中 
通过 “/” 连 接 的 各 项 信息 ， 然 后 将 这 些 信息 以 数组 的 形式 存 入 request_uri_parts 属性 中 。 
function parse http request() { 


Sthis->params = array(); 
$p = $this->request uri parts; 


/* 判 断 数组 $p 是 否 有 索引 为 0 的 元 素 、 该 元 素 值 的 长 度 是 否 为 0、 以 及 该 元 素 值 是 否 不 以 "?" 
开头 ， 若 是 ， 则 将 该 值 存 入 controller 属性 中 。*/ 
if (isset(Sp[0]) && $p[0] gg $p[0] [0] !='?2') 

$this->controller=$p[0]; 


/* 判 断 数 组 $p 是 否 有 索引 为 1 的 元 素 、 该 元 素 值 的 长 度 是 否 为 0、 以 及 该 元 素 值 是 否 不 以 "?" 
开头 ， 若 是 ， 则 将 该 值 存 入 action 属性 中 。*/ 
if (isset(Sp[1]) &é& S$p[1] &é& $p[1] [0] !='?') 

$this->action=$p[1]; 


/* 判断 数组 $p 中 除了 上 述 两 个 元 素 之 外 是 否 还 有 其 他 元 素 , 若是 , 则 将 剩余 的 元 素 存 入 params 
属性 中 */ 
if (isset(Sp[2])) 
$this->params=array slice($p,2); 
return S$this; 


} 


在 parse_http_request 方法 中 ， 我 们 把 request_uri_parts 属性 的 值 赋 给 一 个 局 部 变量 $p， 
然后 判断 变量 $p 的 头 两 个 元 素 是 否 存 在 、 这 两 个 元 素 的 值 的 长 度 是 否 为 零 、 以 及 它们 的 首 
位 是 否 为 半角 问号 。 若 变量 $p 的 头 两 个 元 素 存在 、 它 们 的 值 的 长 度 不 为 零 、 且 首位 不 是 半 
角 问 号 ， 则 将 第 一 个 元 素 的 值 存 入 controller 属性 中 ， 第 二 个 元 素 的 值 存 入 action 属性 中 。 
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最 后 ， 再 判断 是 否 还 存在 其 他 的 元 素 ; 若 有 其 他 的 元 素 ， 则 将 剩余 的 用 户 一 并 存 入 params 
属性 中 。 

至 此 ， 用 户 请 求 分 析 完 毕 。 

在 这 里 还 要 提 一 下 Apache 服务 器 的 URL 改写 模块 。 正 是 由 于 我 们 在 Apache 服务 器 
上 局 用 了 URL 改写 模块 并 在 框架 目录 下 配置 了 分 布 式 配置 文件 (-htaccess) ， 才 实现 了 将 
框架 目录 下 的 index.php 页 面 做 为 框架 入 口 的 目的 。 而 在 框架 目录 下 的 index.php 页 面 中 ， 
通过 实例 化 的 派生 自 KISS_Controller 类 的 Controller 类 ， 调 用 了 该 实例 化 对 象 的 
explode http request 和 parse_http_ request 方法 才 达 到 了 获取 并 解析 用 户 请 求 的 目的 。 

(2) 导入 控制 器 文件 route _ request) 


function route request() { 


// 拼 接 所 需 controller 文件 所 在 路 径 
$controllerfile=$this->controller path.$this->controller.'/'.$this->action. 
a 


/* 判 断 所 需 控制 器 名 称 是 否 包含 非法 字符 ， 以 及 上 面 拼接 的 路 径 下 是 耕 存 在 所 需 的 控制 器 文件 ， 若 
包含 非法 字符 或 文件 不 存在 ， 则 调用 request_not_found 方法 ， 返 回 404 页 面 。*/ 
if (!preg match('#^[A-Za-z0-9 -]+$#',$this->controller) || !file exists 
($controllerfile)) 

$this->request not found('Controller file not found: '.$controllerfile); 


// 拼 接 行为 函数 名 称 


$function="'_'.$this->action; 


/* 判 断 拼接 的 行为 函数 名 称 是 否 包含 非法 字符 、 以 及 该 函数 是 否 存在 ， 若 不 存在 ， 调 用 
request_not found 方法 ， 返 回 404 页 面 。*/ 
if (!preg match('#^[A-Za-z ] [A-Za-2z0-9 —]*$#'",$function) || function exists 
($function)) 

$this->request not found('Invalid function name: '.$function); 


// 导 入 所 需 的 控制 器 文件 


require($controllerfile); 


// 判 断 所 需 的 行为 函数 是 否 存在 ， 若 不 存在 ， 则 调用 request_not_found 方法， 返回 404 页 面 
if (!function exists(Sftunction) ) 
Sthis->request not found('Function not found: '.$function); 


// 通 过 call user func array 函数 将 用 户 通过 URL 传递 的 参数 代入 控制 器 文件 中 的 行为 函 
数 中 

call user func array ($function, $this->params); 

return S$this; 


有 

在 route_request 方法 中 , 使 用 controller path、controller 和 action 属性 拼接 了 控制 器 文 
件 所 在 路 径 ， 并 判断 该 控制 器 文件 是 否 存在 ， 控 制 器 文件 中 是 否 存在 指定 的 行为 函数 ， 若 
存在 则 导入 该 控制 器 文件 ， 若 不 存在 ， 则 返回 404 页 面 。 

(3) 输出 404 页 面 (request_not found) 


function request not found(Smsg=" ") { 
header ("HTTP/1.0 404 Not Found"); 
die('<html><head><title>404 Not Found</title></head> 
<body><h1>Not Found</h1><p>' .$msg.'<p>The requested URL was not found on 
this server.</p> 
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<p>Please go <a href="javascript: history.back(1)">back</a> and try again-</P> 
<hr /><p>Powered By: <a href="http://kissmvc.com">KISSMVC</a></p></body> 
</html>'); 

} 

该 方法 只 在 route request 方法 中 调用 。 当 route request 方法 没有 找到 通过 
explode http request 和 parse_http request 方法 分 离 出 来 的 控制 器 和 行为 时 ， 就 会 调用 
request_not found 方法 ， 返 回 404 页 面 。 

现在 ， 我 们 再 回 过 头 去 看 看 应 用 程序 入 口 index.php 文件 中 创建 的 基于 Controller 类 的 
名 为 $controller 的 对 象 。 我 们 发 现 , 在 创建 这 个 名 为 Scontroller 的 对 象 时 , 指定 了 四 个 参数 : 
控制 器 目录 (APP PATH.'controllers/)、 框 架 目 录 (WEB FOLDER) 、 默 认 控 制 器 (main') 
和 默认 行为 函数 ('index') 。 

当 我 们 加 载 index.php 文件 时 ， 就 自动 加 载 了 “app/controllers/main/” 目 录 下 的 名 为 
index.php 的 文件 中 的 同名 行为 函数 。 该 函数 通过 call_user_func array(0) 函 数 调 用 执行 ， 并 
输出 结果 。 这 就 是 为 什么 我 们 能 看 到 如 图 13-2 所 示 页 面 的 原因 。 

如 果 我 们 在 浏览 器 中 输入 http://www.greatwall.tea/chapter13/users/details/a_florrick， 结 
果 会 是 什么 呢 ? 由 于 并 没有 定义 名 为 users 的 控制 器 ， 名 为 details 的 方法 ， 能 看 到 的 只 能 
是 404 页 面 ( 如 图 13-3 所 示 ) 。 页 面 上 提示 我 们 没有 定义 users/details.php 文件 。 


) 404 NotFound 
oo © DD www.greatwalltea/chapter13/users/detail/a_florrick 
| :3: gg [3) Suggested Sites 1) Web Slice Galley Imported From 正 


Not Found 


Controller file not found: app/controllers/users/detail,php 


The requested URL was not found on this server. 


Please go back and try again. 


图 13-3 未 找到 用 户 请 求 的 方法 和 属性 而 返回 404 错误 


13.2.4 视图 (KISS_View) 


在 上 一 小 节 中 提 到 ， 之 所 以 可 以 看 到 如 图 13-2 所 示 的 页 面 ， 是 因为 我 们 通过 
call_user_func_array() 函 数 调 用 了 “app/controllers/main/” 目 录 下 的 名 为 index.php 的 文件 中 
的 同名 行为 函数 。 现 在 我 们 来 看 看 这 个 函数 的 内 容 : 

<?php 

function index($msg="'Hello World!') { 

Sview = new View (APP PATH.'views/layout.php'); 


Sview->set ('msg', Smsg) > 
Sview->dump (); 


> 
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该 函数 有 一 个 参数 gmsg。 当 用 户 没有 指定 参数 值 时 ， 其 默认 值 为 “Hello World!”。 
在 函数 中 ， 创 建 了 一 个 基于 View 类 的 对 象 ， 并 将 其 命名 为 Sview。 这 个 名 为 $view 的 视图 
对 象 中 引用 了 “app/views/” 目 录 下 名 为 layoutphp 的 视图 文件 ， 该 文件 的 内 容 如 下 : 

<html> 

<head><title><?php echo $GLOBALS['sitename'];?></title></head> 

<body> 

<hl><?php echo $msg;?></h1> 

</body> 

</html> 

在 这 个 视图 文件 中 ， 我 们 除了 输出 在 index.php 文件 中 定义 的 全 局 变量 sitename 之 外 ， 
还 输出 了 一 个 名 为 $msg 的 变量 , 而 这 个 变量 是 通过 $view 对 象 的 set 方法 设置 的 。 在 _index 
行为 属性 的 末尾 ， 使 用 $view 对 象 的 dump 方法 向 用 户 浏 览 器 直接 输出 了 这 个 页 面 。 

这 是 怎么 实现 的 呢 ? 为 了 解答 这 个 问题 , 我 们 还 是 先 来 看 看 这 个 名 为 KISS_View 的 视 
图 类 。 

在 KISS_View 类 中 ,定义 了 两 个 受 保护 的 属性 , 它们 分 别 用 来 存放 模板 文件 路 径 ($file) 
和 模板 文件 里 定义 的 所 有 变量 的 数组 〈$vars) 。 

然后 我 们 在 KISS_View 类 中 定义 了 九 种 方法 ， 实 现 如 下 两 大 类 功能 。 


1. 设置 属性 


在 KISS_View 类 中 ， 定 义 了 三 种 方法 来 设置 受 保护 的 属性 ， 它 们 分 别 是 set、_set 和 
add。 其 中 ，set 方 法 可 以 将 模板 文件 中 定义 的 变量 以 数组 元 素 的 形式 注册 到 $vars 属性 中 ; 
而 __set 方法 则 通过 调用 set 方法 向 $vars 属性 中 注册 任意 变量 ;， add 方法 则 用 来 向 Svars 属 
性 中 注册 模板 文件 中 定义 的 数组 。 

2. 生成 页 面 


在 KISS_View 类 中 ， 我 们 还 定义 了 两 组 六 种 方法 来 生成 页 面 。 它 们 的 内 容 分 别 如 下 
所 示 。 
(1) 页 面 缓 存 
在 该 组 中 的 方法 都 是 将 生成 的 页 面 缓 存在 服务 器 内 存 中 ， 并 返回 用 于 获取 缓存 页 面 的 
句柄 。 我 们 可 以 在 执行 这 些 方法 后 ， 使 用 echo 命令 或 者 print 命令 输出 缓存 的 页 面 。 这 些 
用 于 页 面 缓存 的 方法 分 别 是 fetch、do_fetch 和 do_fetch_str。 其 中 ，fetch 用 来 缓存 $var 属 
性 中 各 元 素 及 其 他 临时 变量 与 模板 页 面 拼 合 后 的 结果 , 并 返回 调用 该 结果 的 句柄 ; do_fetch 
则 用 来 缓存 $var 属性 中 各 元 素 及 其 他 临时 变量 与 模板 页 面 拼合 后 的 结果 ， 并 返回 调用 该 结 
果 的 句柄 ; do_fetch_str 则 用 指定 的 字符 串 覆 盖 已 经 缓存 的 Svar 属性 中 各 元 素 及 其 他 临时 变 
量 与 模板 页 面 的 拼合 结果 ， 并 返回 调用 该 字符 串 的 句柄 。 

(2) 直接 显示 

在 该 组 中 的 方法 都 是 直接 将 生成 的 页 面 传递 给 用 户 。 这 些 用 于 直接 显示 生成 页 面 的 方 
法 分 别 是 dump、do_dump 和 do_dump _str。 其 中 ，dump 用 来 直接 向 用 户 浏览 器 输出 $var 
属性 中 各 元 素 及 其 他 临时 变量 与 模板 页 面 拼 合 后 的 结果 ; do_dump 则 用 来 直接 向 用 户 浏览 
器 输出 Svar 属性 中 各 元 素 及 其 他 临时 变量 与 指定 模板 页 面 拼合 后 的 结果 ;do_dump_str 则 
指定 的 字符 串 覆 盖 已 经 缓存 的 Svar 属性 中 各 元 素 及 其 他 临时 变量 与 模板 页 面 的 拼合 结果 。 
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说 到 底 ，fetcher 组 方法 和 dumper 组 方法 的 差别 就 在 于 fetcher 组 的 方法 缓存 已 生成 的 
页 面 ， 并 返回 调用 已 缓存 页 面 的 句柄 ， 而 dumper 组 的 方法 则 直接 向 用 户 浏览 器 输出 已 生 
成 的 页 面 。 这 个 差别 就 是 通过 ob_start0 和 ob_get clean0) 函 数 来 实现 的 。 前 者 用 来 打开 缓存 
开关 ， 而 后 者 则 提供 调用 缓存 数据 的 句柄 。 

下 面 是 KISS_View 类 的 脚本 。 

abstract class KISS View { 


protected $file="'"'; 
protected $vars=array (); 


// 初 始 化 受 保护 属性 $file 和 $vars 
function construct ($file="'"', $vars="'"') { 
if ($file) 
$this->file = $file; 
if (is array($vars)) 
$this->vars=$vars; 
return S$this; 


下 
// 定 义 魔 术 方法 “set， 通 过 调用 set 方法 来 向 Svar 属性 注册 任意 变量 


function _set($key,$var) { 
return $this->set ($key, $var); 
} 


// 定 义 set 方法 ， 用 来 向 $var 属性 注册 模板 页 面 上 定义 的 变量 
function set($key, $var) { 
$this->vars[$key]=$var; 
return S$this; 


} 


// 定 义 add 方法 ， 用 来 向 $var 属性 注册 模板 页 面 上 定义 的 数组 
function add ($key, $var) { 

$this->vars[$key] []=$var; 
D 


/* 定 义 fetch 方法 ， 用 于 缓存 $var 属性 中 各 元 素 、 临 时 变量 与 模板 页 面 拼合 后 的 结果 ， 
并 返回 调用 该 结果 的 句柄 。 */ 
function fetch($vars='') { 

if (is array ($vars)) 

$this->vars=array merge($this->vars, $vars); 

extract ($this->vars); 

ob start(); 

require($this->file); 

return ob get clean(); 


和 


// 定 义 dump 方法 ， 用 于 直接 向 用 户 浏览 器 输出 $var 属性 中 各 元 素 、 临 时 变量 与 模板 页 面 拼合 
后 的 结果 
function dump($vars="'') { 
if (is array ($vars)) 
$this->vars=array merge ($this->vars, $vars); 
extract ($this->vars); 
require($this->file); 


} 


/* 定 义 do_fetch 方法 , 用 于 缓存 $var 属性 中 各 元 素 、 临时 变量 与 指定 模板 页 面 拼合 后 的 结果 ， 
并 返回 调用 该 结果 的 句柄 。 */ 
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static function do fetch ($file="'',$vars='"') { 
if (is array ($vars)) 
extract ($vars); 
Cb tartn 
require ($file); 
return ob get clean(); 


} 


/* 定 义 do_dump 方法 ， 用 于 直接 向 用 户 浏览 器 输出 $var 属性 中 各 元 素 、 临 时 变量 
与 指定 模板 页 面 拼合 后 的 结果 。 */ 
static function do dump ($file="'',$vars="'') { 
if (is array ($vars)) 
extract ($vars); 
require ($file); 


} 


/* 定义 do_fetch_str 方法 ， 可 用 指定 的 字符 串 获 盖 将 已 缓存 的 $var 属性 中 各 元 素 、 临 时 
变量 与 指定 模板 页 面 拼合 结果 。*/ 
static function do fetch str($str,S$vars='') { 

if (is array($vars)) 

extract ($vars); 

ob start(); 

eval('?>' .$str); 

return ob get clean(); 


/* 定义 do_dump_str 方 法， 直接 向 用 户 浏览 器 输出 用 于 获 盖 已 缓存 的 Svar 属性 中 各 元 素 、 
临时 变量 与 指定 模板 页 面 拼合 结果 的 字符 串 。*/ 
static function do dump str($str,$vars='') { 
if (is_array(Svars) ) 
extract ($vars); 
eval('?>' .$str); 
} 
} 
在 了 解 了 KISS_View 类 的 脚本 后 ， 再 回 过 头 去 看 看 “app/controllers/main/” 目 录 下 的 
名 为 index.php 的 文件 中 的 同名 行为 函数 及 其 调用 的 模板 文件 。 这 个 名 为 _index0 的 函数 带 
有 一 个 参数 Smsg， 其 默认 值 为 “Hello World”。 在 _index0 里 ， 我 们 实例 化 了 派生 自 
KISS_View 类 的 View 类 ， 然 后 调用 了 该 对 象 的 set 方法 将 参数 Smsg 的 值 以 “msg” 为 索引 
存 入 了 对 象 的 Svars 属性 中 。 接着 调用 该 对 象 的 dump 方法 向 用 户 浏览 器 输出 页 面 。 在 dump 
方法 中 ， 我 们 将 $vars 属性 中 的 元 素 通过 extractO 函 数 分 离 出 来 ， 然 后 再 导入 模板 文件 ， 从 
而 实现 $vars 属性 中 的 各 元 素 与 模板 文件 的 拼合 过 程 。 


13.2.5 模型 (KISS_Model) 


就 像 我 们 在 13.1.1 小 节 里 提 到 的 那样 : “在 实际 的 应 用 程序 开发 中 ， 并 不 是 所 有 的 用 
户 请 求 都 需要 调用 模型 或 视图 的 。 有 的 用 户 请 求 涉及 到 的 模型 可 能 非常 复杂 ， 一 个 模型 解 
决 不 了 问题 ， 这 时 就 需要 调用 多 个 模型 ， 而 其 中 肯定 会 有 些 模型 做 为 中 间 件 而 不 需要 调用 
视图 。” 

KISSMVC 框架 的 欢迎 页 面 就 没有 用 到 模型 .用 户 将 请 求 发 送 到 框架 ,该 请 求 被 Apache 
服务 器 按照 分 布 式 配置 文件 中 定义 的 规则 转发 到 了 框架 的 唯一 入 口 。 在 框架 获取 并 解析 了 
用 户 请 求 之 后 ， 通 过 调用 所 需 的 视图 类 来 向 用 户 浏览 器 返回 用 户 请 求 的 数据 。 
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这 一 过 程 中 ， 我 们 并 没有 用 到 模型 。 那 么 模型 到 底 在 什么 情况 下 才 会 使 用 呢 ? 

一 般 来 说 ， 涉 及 到 与 数据 库 的 交互 时 ， 才 会 用 到 模型 。 所 谓 模型 ， 其 实 就 是 关系 型 数 
据 库 中 CRUD 操作 的 对 象 化 。 

在 KISSMVC 框架 中 , KISS_Model 类 是 三 个 核心 抽象 类 中 最 复杂 的 一 个 , 但 是 它 并 不 
难 懂 。 在 这 一 小 节 里 ， 我 们 就 来 分 析 一 下 这 个 模型 类 。 

在 这 个 模型 类 中 一 共 定 义 了 六 个 受 保护 的 属性 ， 它 们 分 别 是 : 主键 名 〈$pkname) 、 
表 名 (S$tablename) 、 数 据 库 句柄 函数 名 ($dbhsfnname) 、 引 号 风格 (SQUOTE STYLE) 、 
数组 压缩 (SCOMPRESS_ARRAY) 和 记录 集 ($rs) 。 

该 类 的 属性 和 初始 化 方法 如 下 : 


abstract class KISS Model { 


protected $pkname; 

protected S$tablename; 

protected $dbhfnname; 

protected $QUOTE STYLE='MYSQL'; // 三 种 可 选 引 号 样式 : MYSQL, MSSQL,ANSI 
protected $COMPRESS ARRAY=true; 

public S$rs = array(); // 存 放 所 有 对 象 属性 变量 


function construct ($pkname="'', $tablename="'', $dbhfnname="'getdbh', $quote_ 
style='MYSQL', 
$compress array=true) { 
$this->pkname=$pkname; // 编 号 自 增长 主键 名 
$this->tablename=$tablename; / /数据库 表 名 
$this->dbhfnname=$dbhfnname; // 数 据 库 句柄 函数 名 
Sthis->QUOTE STYLE=$quote_style; // 拼 接 SQL 语句 时 ， 各 字段 名 前 后 使 用 的 引号 风格 
$this->COMPRESS ARRAY=$compress_array; // 绑 定 值 到 SQL 语句 中 各 字段 名 时 ， 某 
值 为 数组 ， 是 否 压 缩 


} 
// 以 下 为 模型 类 的 各 种 方法 ， 暂 略 
除 此 之 外 , 模型 类 中 还 定义 四 种 基础 CRUD 方法 、 四 种 高 级 查询 方法 和 七 种 辅助 方法 ， 
共计 15 个 方法 。 
现在 我 们 就 来 看 看 这 些 方法 。 
1. 基础 CRUD 方 法 


我 们 之 前 说 过 ， 模 型 类 的 目的 是 与 数据 库 交 互 ， 完 成 数据 的 存储 和 查询 任务 ， 而 这 些 
任务 就 会 涉及 到 CRUD 操作 。KISSMVC 框架 的 模型 类 一 共有 四 个 基础 的 CRUD 方法 , 它 
们 分 别 是 create、retrieve、update 和 delete， 分 别 用 来 向 数据 库 添加 一 条 记录 、 从 数据 库 中 
提取 一 条 记录 、 更 新 数据 库 中 的 一 条 指定 的 记录 以 及 把 一 条 指定 记录 从 数据 库 中 删除 。 

编写 这 四 个 基础 CRUD 方法 的 思路 是 ， 先 通过 辅助 方法 (如 set) 将 需要 添加 、 查 询 、 
更 新 或 删除 的 数据 记录 的 字段 名 和 值 存 入 对 象 私有 属性 $rs 中 ， 然 后 通过 遍历 $rs 中 的 各 元 
素来 拼接 SQL 语句 ， 最 后 执行 拼接 好 的 SQL 语句 ， 并 返回 对 象 。 

下 面 ， 我 们 先 来 看 看 create 方法 的 脚本 : 
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的 时 候 数据 库 中 的 某 个 字段 的 值 可 能 会 是 一 个 非 标准 变量 ， 这 时 就 要 通过 合理 使 用 


// 以 下 方法 用 于 向 数据 库 中 添加 一 条 记录 
function create () { 
// 获 取 数据 库 操 作 句 柄 
$dbh=$this->getdbh (); 


// 将 主键 名 赋值 给 局 部 变量 $pkname 


$pkname=$this->pkname; 


// 定 义 两 个 空 字符 串 用 来 拼接 查询 语句 
$s1=$s2=""; 


/* 将 私有 属性 $rs 中 的 各 元 素 的 索引 用 半 脚 逗号 连接 存 入 局 部 变量 $s1， 
然后 将 对 应 数量 的 "2" 用 半 脚 逗号 连接 存 入 局 部 变量 $s2 中 */ 
foreach ($this->rs as $k => $v) 

if ($k!=$pkname || S$v) { 
$s1 .= ','.$this->enquote ($k); 
2 

} 
// 拼 接 SQL 语句 
$sql = 'INSERT INTO ' .Sthis->enquote (Sthis->tablename).' ('.substr($s1,1).') 
VALUES (' .substr($s2,1) .') 7 


// 准 备 执行 SQL 语句 
$stmt = $dbh->prepare ($sql); 


// 将 属性 $rs 中 各 元 素 的 值 绑 定 到 SQL 语句 中 定义 的 各 字段 名 
$i=0; 
foreach (Sthis->rs as $k => $v) 
if ($k!=$pkname || $v) 
$stmt->bindValue (++$i,is scalar($v) ? 
$v : ($this->COMPRESS ARRAY ? gzdeflate (serialize 
($v)) : serialize($v)) ); 


// 执 行 准备 好 的 SQL 语句 


$stmt->execute(); 


/* 通过 rowCount () 函数 获取 执行 此 SQL 语句 后 受到 影响 的 记录 数量 ， 若 该 函数 返回 FALSE， 
则 添加 失败 。 
此 时 ， 直 接 返 回 FALSE， 退 出 该 方法 。 */ 
if (!$stmt->rowCount () ) 
return false; 


// 获 取 已 添加 记录 的 序号 ， 并 将 其 存 入 属性 Spkname 中 
Sthis->set ($pkname, $dbh->lastInsertId()); 


// 返 回 对 象 
return S$this; 


| 
在 create 方法 的 脚本 中 ， 我 们 就 是 按照 上 文中 提 到 的 思路 编写 的 。 需 要 注意 的 是 ， 有 


serialize() 和 unserializeO) 函 数 来 简化 数据 库 的 表 结 构 。 在 上 面 的 脚本 中 , 我 们 在 绑 定 属性 Srs 
中 各 元 素 的 值 到 SQL 语句 中 各 字段 名 时 , 根据 属性 SCOMPRESS_ARRAY 的 值 来 判断 是 否 
压缩 非 标 变量 序列 化 后 的 值 。 这 样 一 来 ， 数 据 库 中 各 字段 里 也 可 以 存储 数组 、 对 象 之 类 的 
非 标 变量 值 了 。 
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下 面 再 来 看 看 如 何 查询 数据 库 中 符合 条 件 的 数据 记录 。 
// 以 下 方法 用 于 在 数据 库 中 指定 主键 值 的 一 条 数据 记录 ， 参 数 为 主键 值 


function retrieve (Spkvalue) { 


// 获 取 数据 库 操 作 句柄 
$dbh=$this->getdbh(); 


// 拼 接 SQL 语句 
$sql = 'SELECT * FROM ' .$this->enquote ($this->tablename) .' WHERE ' .$this-> 
enquote (Sthis->pkname) . "=2"7 


// 准 备 执 行 SQL 语句 
$stmt = $dbh->prepare ($sql); 


// 绑 定 主键 值 到 SQL 语句 中 指定 的 主键 名 
$stmt->bindValue (1, (int)Spkvalue) 


// 执 行 SQL 语句 


$stmt->execute(); 


// 将 SQL 语句 执行 结果 以 数组 的 形式 存 入 局 部 变量 $rs 中 ， 各 元 素 的 索引 为 其 值 对 应 的 字段 名 
Srs = $stmt->fetch (PDO: :FETCH ASSOC) 7 


/* 如 果 局 部 变量 $rs 不 为 定 ， 则 遍历 该 数组 ， 以 便 还 原 之 前 被 压缩 的 非 标 变量 值 ， 
并 将 它们 存 入 属性 $rs 中 的 对 应 元 素 里 。*/ 
LE (SESY 
foreach (Srs as $key => $val) 
if (isset($this->rs[$key])) 
$this->rs[$key] = is scalar($this->rs[$key]) ? 
$val : unserialize($this->COMPRESS ARRAY ? gzinflate ($val) : $val); 


// 返 回 对 象 
return Sthis7 


} 


在 retrieve 方法 的 脚本 中 ， 我 们 定义 了 如 何 从 数据 库 中 获取 指定 主键 值 的 数据 记录 。 
获取 此 类 数据 记录 的 SQL 语句 较为 简单 ， 拼 接 SQL 语句 时 也 很 轻松 ， 只 要 注意 WHERE 
子 名 的 内 容 就 好 。 在 将 查询 到 的 数据 存 入 属性 $rs 中 之 后 ,我 们 遍历 数组 ,还原 可 能 被 压缩 
或 序列 化 的 非 标 变量 值 。 最 后 ， 返 回 对 象 ， 以 便 调 用 对 象 的 其 他 方法 。 

下 面 两 个 方法 与 create 和 retrieve 方法 类 似 , 大 家 可 以 根据 注释 自己 揣摩 , 我 们 就 不 再 
歼 述 了 。 

// 以 下 方法 用 于 更 新 数据 库 中 指定 主键 值 的 一 条 数据 记录 


function update () { 


// 获 取 数 据 库 操作 句柄 
$dbh=$this->getdbh (); 


// 定 义 局 部 变量 $s， 用 于 拼接 SQL 语句 。 这 一 点 与 create 方法 中 的 局 部 变量 $s1 和 $s2 类 似 


$s=""; 
foreach (Sthis->rs as $k => $v) 
$s .= ','.$this->enquote ($k).'=2"'; 


$s = substr($s,1); 
$sql = "UPDATE ' -Sthis->enquote (Sthis->tablename) - 
' SET '.$s.' WHERE ' .Sthis->enquote (Sthis->pkname) .'=?277 
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// 准 备 执行 SQL 语句 
$stmt = $dbh->prepare ($sql); 


// 绑 定 UPDATE 主 句 中 的 字段 名 和 值 
$i=0; 
foreach ($this->rs as $k => $v) 
$stmt->bindValue (++$i,is scalar(Sv) ? 
$v : (Sthis->COMPRESS ARRAY ? gzdeflate (serialize ($v)) : 
serialize ($v)) ); 


// 绑 定 WHERE 子 句 中 的 字段 名 和 值 


$stmt->bindValue (++$i, $this->rs[$this->pkname]); 


// 执 行 SQL 语句 ， 并 返回 执行 结果 
return $stmt->execute(); 


. 
// 以 下 方法 用 于 删除 数据 库 中 指定 主键 值 的 一 条 数据 记录 


function delete() { 


// 获 取 数据 库 操作 句柄 
$dbh=$this->getdbh(); 


// 拼 接 SQL 语句 
$sql = 'DELETE FROM ' .$this->enquote (Sthis->tablename) .' WHERE ' .Sthis-> 
enquote ($this->pkname) . "=2 17 


// 准 备 执行 SQL 语句 
$stmt = $dbh->prepare ($sql); 


// 绑 定 WHERE 子 句 中 的 字段 名 和 值 
$stmt->bindValue (1, $this->rs[$this->pkname]); 


// 执 行 SQL 语句 ， 并 返回 执行 结果 
return $stmt->execute(); 


i’ 
2 高 级 查询 方法 


在 基础 CRUD 方法 中 的 查询 方法 (retrieve) 固然 很 妙 , 但 是 却 只 能 根据 指定 的 主键 值 
来 查询 一 条 数据 记录 ， 实 在 不 能 完全 满足 我 们 的 查询 需求 。 这 时 可 以 使 用 模型 类 中 定义 的 
四 种 高 级 查询 方法 来 丰富 我 们 的 查询 方式 ， 它 们 分 别 是 exists、retrieve_one、 retrieve_many 
和 select， 分 别 用 来 判断 某 条 数据 记录 是 否 存在 、 查 询 一 条 满足 指定 条 件 的 数据 记录 、 查 
询 多 条 满足 条 件 的 数据 记录 并 返回 这 些 记录 的 所 有 字段 和 查询 多 条 满足 条 件 的 数据 记录 并 
返回 这 些 记录 的 指定 字段 。 

下 面 我 们 就 来 看 看 这 四 种 高 级 查询 方法 的 脚本 : 

/* 以 下 方法 用 来 查询 某 条 数据 是 否 存在 ， 若 存在 ， 则 根据 参数 Scheckdb 的 值 返 回 TRUE 或 1; 

若 不 存在 ， 则 返回 FALSE。*/ 

function exists($checkdb=false) { 

if ((int)$this->rs[$this->pkname] < 1) 
return false; 


if (!$checkdb) 
return true; 


// 获 取 数 据 库 操作 句柄 
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$dbh=$this->getdbh(); 


// 拼 接 SQL 语句 

$sql = "SELECT 1 FROM ' .Sthis->enquote (Sthis->tablename) - 
' WHERE ' .Sthis->enquote (Sthis->pkname) ."='".$this->rs[$this-> 
Pkname] -"”" "7 


// 执 行 SQL 语句 ， 并 将 执行 结果 存 入 局 部 变量 $result 中 
$result = $dbh->query ($sql) ->fetchAll (); 


// 返 回 $result 中 记录 的 数量 
return count ($result); 


bi 


/* 以 下 方法 用 于 查询 数据 库 一 条 满足 指定 条 件 的 数据 记录 。 若 存在 多 条 满足 此 指定 条 件 的 记录 ， 
则 返回 第 一 条 记录 。*/ 


function retrieve one($wherewhat="'', $bindings="'') { 


// 获 取 数据 库 操作 句柄 
$dbh=$this->getdbh(); 


// 判 断 参 数 Sbindings 是 否 为 标准 变量 。 若 是 ， 则 将 其 以 数组 的 形式 存 入 局 部 变量 $bindings 中 
if (is scalar($bindings)) 
Sbindings= trim($bindings) ? array(Sbindings) : array() 7 


// 拼 接 SQL 语句 
$sql = "SELECT * FROM ' .S$this->enquote (Sthis->tablename) ; 


// 判 断 参数 Swherewhat 是 否 设置 。 若 设置 ， 则 将 其 添加 到 SQL 语句 末尾 
if ($wherewhat) 


$sql .= ' WHERE '.S$wherewhat; 
// 在 SQL 语句 末尾 添加 LIMIT 子 句 ， 以 便 在 多 条 记录 满足 指定 条 件 时 只 输出 第 一 条 满足 条 件 的 记录 
SET ER 


// 准 备 执行 SQL 语句 
$stmt = $dbh->prepare ($sql); 


// 读 取 局 部 变量 $bindings 的 值 ， 并 将 它们 与 WHERE 子 句 中 指定 的 各 字段 名 绑 定 
$i=0; 
foreach ($bindings as $v) 

$stmt->bindValue (++$i, $v); 


// 执 行 SQL 语句 


$stmt->execute(); 


// 将 查询 结果 存 入 局 部 变量 Srs 中 
Srs = $stmt->fetch (PDO: :FETCH ASSOC) > 


/* 如 果 局 部 变量 $rs 不 为 空 ， 则 遍历 该 数组 ， 以 便 还 原 之 前 被 压缩 的 非 标 变量 值 ， 
并 将 它们 存 入 属性 Srs 中 的 对 应 元 素 里 。*/ 
if (1Szs) 
return false; 
foreach ($rs as $key => $val) 
if (isset($this->rs[$key])) 
Sthis->rs[Skey] = is scalar(Sthis->rs[Skey]) ? 
$val : unserialize (Sthis->COMPRESS ARRAY ? gzinflate 
($val) : $val); 
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// 返 回 对 象 


return S$this; 


// 以 下 方法 用 于 查询 满足 指定 条 件 的 所 有 记录 。 与 retrieve_one 方法 唯一 不 同 的 是 ， 
retrieve_one 方法 在 拼接 SQL 语句 时 ， 添 加 了 IIMIT 子 句 , 而 retrieve many 方法 则 没有 。*/ 
function retrieve many($wherewhat="',$bindings="') { 
$dbh=$this->getdbh(); 
if (is scalar($bindings)) 

$bindings=trim($bindings) ? array($bindings) : array(); 
$sql = "SELECT * FROM '.$this->tablename; 
if ($wherewhat) 


$sql .= ' WHERE '.S$wherewhat; 
$stmt = $dbh->prepare ($sql); 
$i=0; 


foreach ($bindings as S$v) 
$stmt->bindValue (++$i, $v); 
$stmt->execute(); 
$arr=array (); 
$class=get class($this); 
while (Srs = $stmt->fetch(PDO: :FETCH ASSOC)) { 
$myclass = new $class(); 
foreach (Srs as $key => $val) 
if (isset ($myclass->rs[$key])) 
$myclass->rs[$key] = is scalar($myclass->rs[$key]) ? 
Sval : unserialize(Sthis->COMPRESS ARRAY ? 
gzinflate ($val) : $val); 
$arr[]=$myclass; 
} 
return S$arr; 


} 


/* 以 下 方法 用 于 查询 满足 指定 条 件 的 所 有 记录 (指定 字段 〉。 该 方法 与 retrieve_many 类 似 ， 
只 是 在 拼接 SQL 语句 和 绑 定 SQL 语句 中 的 字段 名 和 值 时 , 加 入 了 参数 $selectwhat 中 指定 的 内 容 。*/ 
function select ($selectwhat="'*', $wherewhat="'', $bindings="', $pdo_fetch mode 
=PDO: :FETCH ASSOC) { 

$dbh=$this->getdbh(); 

if (is_ scalar ($bindings)) 

$bindings=trim($bindings) ? array ($bindings) : array(); 
$sql = 'SELECT '.$selectwhat.' FROM '.S$this->tablename; 
if (S$wherewhat) 


$sql .= ' WHERE '.S$wherewhat; 
$stmt = $dbh->prepare ($sql); 
$i=0; 


foreach ($bindings as S$v) 
$stmt->bindValue (++$i, $v); 

$stmt->execute(); 

return $stmt->fetchAll($pdo fetch mode); 


3. 辅助 方法 

除开 上 述 八 种 主要 的 CRUD 操作 方法 之 外 ，KISSMVC 框架 的 模型 类 还 为 我 们 提供 了 
七 种 辅助 方法 ， 用 于 设置 和 获取 某 些 属性 的 值 。 它 们 是 set、get、_ set、_ get、getdbh、 
enquote 和 merge， 分 别 用 来 设置 属性 Srs 元 素 、 获 取 属性 $rs 中 指定 元 素 的 值 、 获 取 数 据 库 
操作 句柄 、 定 义 SQL 语句 引号 风格 和 合并 数组 到 Srs 属性 中 。 

下 面 我 们 就 来 看 看 这 七 种 方法 的 脚本 : 
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// 以 下 方法 用 于 获取 $rs 属性 中 指定 元 素 的 值 
function get ($key) { 

return $this->rs[$key]; 
} 


// 以 下 方法 用 于 设置 $rs 属性 中 指定 元 素 的 值 
function setl($key, $val) { 
if (isset($this->rs[$key])) 
$this->rs[$key] = $val; 
return S$this; 


b 


// 同 get 方 法 

function get($key) { 
return Sthis->get ($key); 

| 


// 同 set 方法 

function set ($key, S$val) { 
return $this->set ($key, $val); 

} 


// 以 下 方法 通过 call_user_func() 函数 调用 用 户 自 定义 的 数据 库 连 接 函 数 
protected function getdbh() { 
return call user func($this->dbhfnname); 


} 
// 以 下 方法 用 于 定义 在 拼接 SQL 语句 中 ， 使 用 的 引号 风格 


protected function enquote($name) { 
if ($this->QUOTE STYLE=="MYSQL') 


return '‘'.$name.''; 

elseif ($this->QUOTE STYLE=="MSSQL') 
roturn M1 -name “1 

else 
return '"'.$name.'™"'; 


} 
// 以 下 方法 用 于 向 $rs 属性 添加 元 素 


function merge (Sarr) { 
if (!is array(Sarr) ) 
return Sthis7 
foreach ($arr as S$key => $val) 
if (isset($this->rs[$key])) 
$Sthis->rs[$key] = $val; 
return S$this; 


13.2.6 ”使 用 控制 器 操控 模型 和 视图 


在 上 一 小 节 里 , 我 们 提 到 , 在 模型 类 的 辅助 方法 getdbh 中 , 我 们 使 用 了 call_user func() 


函数 获取 用 户 自 定义 的 获取 数据 库 操作 句柄 的 函数 。 这 样 就 赋予 了 给 我 们 更 大 的 选择 数据 
库 类 型 的 权利 。 如 果 读 者 不 想 自己 定义 获取 数据 库 连 接 的 函数 ，KISSMVC 框架 也 为 我 们 


写 好 了 一 个 函数 ， 直 接 修改 一 下 就 可 以 用 了 。 


打开 KISSMVC 框架 目录 下 的 index.php 文件 ， 找 到 如 下 内 容 : 


//Database 
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========= ============ ===== 


function getdbh() { 
if (!isset($GLOBALS ["dqbh'])) 
try { 
$GLOBALS['dbh'] = new PDO('sqlite:'.APP PATH.'db/kissmvc.sqlite'); 
//$GLOBALS['dbh'] = new PDO('mysql:host=localhost;dbname=kissmvc', 
'username', 'password'); 


} catch (PDOException $e) { 
die('Connection failed: '.$e->getMessage()); 
} 
return $GLOBALS['dbh']; 
. 
*/ 
如 果 需 要 启用 该 getdbh0) 函 数 ， 请 移 除 该 函数 前 后 的 注释 符 (/* .….*/) 即 可 。 若 需要 使 
用 SQLite 数据 库 ， 则 需要 注 明 数据 库 文件 的 路 径 。 若 需要 使 用 MySQL 数据 库 ， 则 需要 注 
释 掉 SQLite 数据 库 的 那 一 行 ， 并 将 MySQL 数据 库 对 应 的 那 一 行 前 面 的 注释 符 去 掉 ， 然 后 
修改 主机 名 、 数 据 库 名 、 用 户 名 和 密码 。 在 这 里 提 一 句 ， 下 面 的 例子 中 使 用 的 是 MySQL 
数据 库 ， 主 机 名 为 localhost， 数 据 库 名 为 kissmvc， 用 户 名 为 kissmvc， 密 码 为 Passw0rd。 
随后 ， 我 们 还 需要 启动 自动 加 载 类 文件 功能 。 在 KISSMVC 框架 目录 下 的 index.php 
文件 中 ， 找 到 如 下 脚本 : 


//Assumes Model Classes start with capital letters and Helpers start with 
lower case letters 
/* 
function autoload($classname) { 
$a=$classname [0]; 
if ($a >= 'A' && $a <='2Z') 
require once (APP PATH.'models/'.$classname.' .php'); 
else 
require once (APP_ PATH.'helpers/' .$classname.' .php'); 


} 
*/ 
移 除 _autoload(0) 函 数 前 后 的 注释 符 〈/* ...*/) 即 可 。 需 要 注意 的 是 ， 如 果 启 用 自动 加 
载 类 文件 功能 ， 可 以 免 去 我 们 在 定义 模型 、 控 制 器 和 视图 时 再 使 用 require0 和 include0) 函 
数 。 但 是 ， 模 型 类 的 命名 必须 是 大 写字 母 开头 ， 而 框架 助手 类 的 命名 必须 是 小 写字 母 开 头 。 
和 否则， 框架 会 找 不 到 模型 和 框架 助手 而 出 错 。 
现在 做 好 了 准备 工作 ， 来 一 起 看 看 怎么 使 用 控制 器 操控 模型 吧 。 
假设 kissmvc 数据 库 中 有 一 张 名 为 users 的 表 ， 表 中 有 如 下 几 个 字段 : uid、username、 
包 llname 、gender 和 password。 大 家 可 以 自行 在 MySQL 数据 库 中 创建 这 样 一 个 数据 表 。 
然后 进入 kissmvc 框架 目录 ， 在 应 用 目录 (app) 下 的 model 文件 夹 里 创建 一 个 名 为 
user.php 的 文件 。 打 开 该 文件 ， 然 后 输入 如 下 脚本 : 
<?php 
class User extends Model { 
function User() { 
Parent:: construct('uid','user','getdbh'); 


Sthis->rs['uid'] = :7 
Sthis->rs['username'] = "7 
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Sthis->rs["password'"'] = 
Sthis->rs["fullname'] = 
Sthis->rs["gender'] = "7 
Sthis=>Fsilereateda dt TY 


} 
} 


= 

这 段 脚 本 定义 了 派生 自 Model 类 的 User 类 。 这 就 意味 着 ， 一 个 模型 对 应 一 张 数 据 表 。 
由 于 我 们 需要 使 用 自动 加 载 类 文件 的 功能 ， 所 以 在 定义 类 时 ， 注 意 首 字母 大 写 ， 这 样 就 可 
以 在 使 用 类 时 不 引用 相应 的 类 文件 。 

在 这 段 脚 本 中 ， 首 先 初始 化 了 User 类 的 父 类 Model， 指 定 了 数据 表 的 主键 名 、 表 名 和 
数据 库 操控 句柄 函数 名 。 其 中 ，parent 关键 字 指 代 当 前 类 (User) 的 父 类 (Model) ; 接着 
定义 了 User 类 的 $rs 属性 , 将 数据 库 表 各 字段 的 名 称 做 为 数组 元 素 索引 存 入 S$rs 属性 中 。 这 
样 我 们 就 定义 好 了 一 个 模型 。 

接 下 来 , 我 们 在 应 用 目录 下 的 controllers 子 文件 夹 下 , 新 建 一 个 名 为 user 的 子 文件 夹 ; 
然后 在 其 中 新 建 一 个 名 为 add.php 的 文件 。 打 开 该 文件 ， 输 入 如 下 脚本 : 

<?php 

function add($action="''){ 

Smsg = "'"'; 

$view = new View (APP_PATH.'views/adduser .php'); 
if ($action == "do'){ 


// 获 取 用 户 提交 的 数据 。 

$username $_POST['username']; 
$fullname $_POST['fullname']; 
S$gender = POST['gender']; 
Spassword $ POST['password']; 
$confirm = $ POST['confirm']; 


We 


// 验 证 用 户 提交 的 数据 。 
if(strlen($username) < 1 || strlen($fullname) < 1 11 
strpos ($fullname, ' ') == FALSE || strlen($password) < 10 11 
Spassword <> $confirm) { 
Smsg = "Please ensure the correctness of your input."; 
} else { 
// 若 用 户 提交 的 数据 符合 要 求 ， 则 将 它们 存 入 数据 库 。 
$user = new User(); 
$user->set('username',$ POST['username']); 
$user->set('fullname',$ POST['fullname']); 
$user->set('gender',$ POST['gender']); 
$user->set('password',$ POST['password']); 
$user->set ('created dt',date('Y-m-d')); 
$user->create () : 
Suid = (string) Suser->get('"uid') 7 
if($uid){ 
$msg = "This user has been added. The user ID is {$uid}."; 


} else { 
Smsg = "Failed to add this user."; 
} 
} 
} 
Sview->set ('msg', $msg); 
Sview->dump (); 


区 
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在 这 段 脚本 中 ， 定 义 了 一 个 名 为 add0 的 函数 。 该 函数 与 addphp 同名 ， 会 被 引用 到 
KISSMVC 框架 目录 下 index.php 文件 中 创建 的 Scontroller 对 象 中 。 在 _add0 函 数 中 , 实例 化 
了 View 类 ， 引 用 了 位 于 应 用 目录 下 views 子 目 录 中 的 名 为 adduser.php 模板 文件 。 然 后 获 
取 并 分 析 用 户 提交 的 数组 。 若 用 户 提交 的 数据 符合 要 求 ， 则 将 用 户 提交 的 数据 绑 定 到 $user 
对 象 的 $rs 属性 中 ， 然 后 再 使 用 Suser 对 象 的 create 方法 将 绑 定好 的 数据 存 入 数据 库 。 随 后 ， 
再 使 用 $user 对 象 的 get 方法 获取 新 添加 记录 的 uid 字段 的 值 ， 输 出 给 用 户 。 

最 后 ， 将 生成 的 Smsg 信息 绑 定 到 视图 文件 中 的 Smsg 变量 ， 输 出 模板 页 面 到 用 户 浏览 器 。 

现在 , 我 们 在 浏览 器 中 访问 KISSMVC 框架 , 在 地 址 栏 中 的 地 址 末尾 加 上 “user/add”， 
然后 回 车 ， 会 看 到 如 图 13-4 所 示 的 页 面 。 


DD Add User - KISSMVC -Sin x We 


4 © DD wwwgreatwalltea/chapterl3/user/add 
应用 [3 Suggested Stes Web Slice Gallery DD Imported From EE 


图 13-4 添加 用 户 


当 我 们 填写 完 表单 之 后 ， 单 击 “Add This User” 按 钮 添加 该 用 户 。 如 果 数 据 库 连接 正 
常 且 填写 的 信息 符合 要 求 ， 会 看 到 如 图 13-5 所 示 的 页 面 。 


€ 3 CC ID wwwgreatwalltea/chapter13/user/add/do 
洲 应 用 回 Suggested Stes 门 WebSlice Gallery 加 Imported From IE 


Add User 


Username 

Full Name 

Gender Male 加 

Pasword 

Confirm Password 
Add This User 


This user has been added. The user ID is 4. 


图 13-5 添加 用 户 成 功 


这 样 看 来 ， 我 们 成 功 地 使 用 控制 器 操控 模型 完成 了 向 数据 库 添 加 数据 的 任务 。 
看 到 这 里 ， 有 的 同学 可 能 会 说 ， 这 么 写 程序 累 不 累 啊 ， 又 是 控制 器 ， 又 是 模型 的 ， 结 
果 完 成 的 还 是 一 个 很 简单 的 任务 。 我 只 能 说 : “兄弟 ， 你 太 不 淡定 了 ! ” 
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还 记得 我 们 在 本 章 的 开头 说 过 为 什么 要 使 用 MVC 框架 吗 ? 其 中 最 重要 的 一 名 就 是 
“由 于 这 三 个 部 分 〈 指 控制 器 、 模 型 和 视图 ) 相互 独立 ， 我 们 对 其 中 某 个 部 件 的 修改 完全 不 
会 对 其 他 两 个 部 件 造成 影响 。” 为 了 验证 这 个 说 法 ， 现 在 就 在 应 用 目录 下 的 views 文件 夹 
下 再 新 建 一 个 名 为 adduser-bootstrap php 的 文件 , 使 用 Bootstrap UI 框架 来 重 写 “ 添 加 用 户 ” 
的 页 面 。 然 后 打开 controllers/user 目录 下 的 add.php 文件 ， 找 到 如 下 内 容 : 

Sview = new View (APP PATH.'views/adduser .php'); 

将 其 奉 换 成 : 


Sview = new View(APP PATH.'views/adduser-admin.php'); 


然后 再 加 载 一 下 这 个 页 面 ， 会 发 现 ， 这 个 页 面 的 布局 变 成 了 如 图 13-6 所 示 的 样子 。 


Add User 


Full Name 


Gender 


Ppassword 


Confirm Password 


图 13-6 添加 用 户 ( 新 布局 ) 
而 当 用 户 添 加 成 功 后 的 页 面 则 如 图 13-7 所 示 。 


| ET- 
€ 3 CD wworeavwalltea/ chapreri /ser/dd/do 
Sgeted Sees Web Sice Guley DS ingored homE 


Add User 


Username 
Fall Name 
Gender 


Password 


Confirm Password 


图 13-7 添加 用 户 成 功 ( 新 布局 ) 
这 里 ， 我 们 只 对 模板 文件 进行 了 修改 。 为 了 说 明 模板 文件 是 相互 独立 的 ， 在 应 用 目录 
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的 views 文件 夹 下 新 建 了 一 个 名 为 adduser-bootstrap.php 的 模板 文件 ， 然 后 修改 了 
controllers/user 文件 夹 下 的 add.php 文件 。 读 者 完全 可 以 把 adduser-bootstrap.php 的 内 容 复 
制 到 adduserphp 模板 文件 中 。 这 样 一 来 , 完全 不 用 修改 controllers/user 文件 夹 下 的 add.php 
文件 ， 就 可 以 实现 更 换 页 面 布 局 的 目的 。 

使 用 MVC 框架 编写 的 脚本 维护 起 来 真是 不 太 方便 啊 ， 你 觉得 呢 ? 


13.3 ”扩充 框架 : 基于 MVC 的 记 账 工具 


还 记得 我 们 在 第 8 一 10 章 里 编写 的 记 账 工具 吗 ? 那个 记 账 工具 虽然 是 基于 对 象 编 程 的 
产物 ， 但 却 没有 使 用 任何 的 设计 模式 ， 所 有 的 属性 和 方法 都 写 到 了 一 个 类 里 。 这 样 不 利于 
后 续 对 脚本 进行 维护 ， 同 时 也 很 难 对 工具 的 用 户 界面 进行 重新 设计 。 

在 本 小 节 里 ， 将 使 用 KISSMVC 框架 来 重新 编写 我 们 的 记 账 工具 ， 将 记 账 工具 的 业务 
逻辑 、 数 据 库 读 写 和 用 户 界面 呈现 分 离开 来 。 同 时 ， 将 会 为 这 个 记 账 工具 添加 一 些 新 的 功 
能 ， 并 通过 对 用 户 更 加 友好 的 界面 元 素 加 以 呈现 。 在 记 账 工具 的 模板 部 分 ， 我 们 将 使 用 基 
于 Bootstrap UI 框架 的 一 套 模板 和 站。 大 家 也 可 以 了 解 一 下 如 何 使 用 Bootstrap 这 一 当前 十 分 
流行 的 用 户 界面 框架 。 


13.3.1 数据 规划 


按照 第 8 章 规划 的 功能 , 将 在 本 小 节 里 完成 对 基于 MVC 的 记 账 工具 的 数据 规划 工作 。 
表 13-1 规划 了 项 目 名 称 、 数 据 库 类 型 、 数 据 库 连接 数据 和 主要 功能 。 


表 13-1 基本 数据 规划 


项 目 数 据 
项 目 名 称 PennyCounts — Every Penny Counts! 
项 目 架构 KISSMVC 
数据 库 类 型 MySQL 

IE 
ss 数据 库 名 kissmve 
各 失语 二 作 用 户 名 kissmve 
密码 PasswOrd* 


(1) 用 户 登 录 、 注 销 与 验证 

(2) 逐条 /批量 添加 收入 记录 

(3) 逐条 /批量 添加 支出 记录 
主要 功能 (4) 输出 收入 /支出 列表 

(5) 修改 指定 的 收入 和 支出 记录 

(6) 删除 指定 的 收入 和 支出 记录 

(7) 输出 分 月 统计 报表 


晶 你 可 以 从 Bootstrap 的 官方 网 站 (http://getbootstrap.com) 上 获取 源 代码 。 从 Bootswatch 网 站 
(http://bootswatch.com) 上 获取 开源 的 Bootstrap UI 框架 主题 ， 关 于 如 何 使 用 Bootstrap UI 框架 ， 请 参考 
Bootstrap 官方 网 站 上 的 文档 说 明 。 本 书 中 使 用 的 Bootstrap UI 框架 主题 是 Bootswatch 网 站 上 提供 的 名 为 
Readable 主题 (http://bootswatch.com/readable/) 。 
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在 定义 了 基本 信息 之 后 ， 需 要 在 数据 库 中 建立 若干 张 数据 表 ， 用 于 存放 用 户 、 收 入 记 
录 和 消费 记录 数据 。 具 体 规划 如 表 13-2 所 示 。 
表 13-2 数据 表 规划 


字 段 名 数据 类 型 备注 
uid INT(1) 主键 、 非 空 、 自 增长 


Username VARCHAR (25) 
follname VARCHAR (60) 
用 户 表 (user) gender VARCHAR (10) 


password VARCHAR (25) 
avatar VARCHAR (255) 
created_dt DATE 


NT OD) 


description VARCHAR (128) 
amount FLOAT (8, 2) 
add dt DATE 

ATE 


二 | 马 | 刀 | 局 | 了 | 也 | 马 


收入 记录 表 (income) 
ao 
ww AT GD | 
aa 
人 


接 下 来 ， 为 每 个 功能 规划 一 下 路 由 、 控 制 器 、 模 型 和 视图 文件 ， 如 表 13-3 所 示 。 
表 13-3 ”详细 数据 规划 


支出 记录 表 (expense) 


be 


项 目 数 据 

控制 台 

路 E main/dashboard 

可 选 参数 无 

控制 器 app/controllers/main/dashboard.php 

模型 app/models/incomes.php 和 app/models/expenses.php 

视图 app/views/dashboards php 

用 户 登 录 和 验证 

路 main/login/action/msg_id 
(1) action ， 表 示 用 户 是 否 单 击 “ 登 录 ” 按 钮 ， 其 可 选 值 如 下 : 
do， 表 示 用 户 已 单 击 “ 登 录 ” 按 钮 。 

可 选 参数 CNULL) ， 表 示 用 户 未 单 击 “ 登 录 ” 按 钮 。 

国 (2) msg_ id， 表示 用 户 登录 结果 ， 其 可 选 值 如 下 : 

301， 表 示 用 户 输入 的 信息 不 符合 要 求 。 
401， 表 示 数 据 库 连 接 失 败 ， 无 法 登录 
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项 目 数 据 

控制 器 app/controllers/main/login.php 

模型 app/models/user.php 

视图 app/views/login php 

用 户 注册 

路 由 main/register/action/msg_id 
(1) action， 表 示 用 户 是 否 单 击 “ 注 册 ” 按 钮 ， 其 可 选 值 如 下 : 
do， 表 示 用 户 已 单 击 “ 注 册 ” 按 钮 。 
(NULL) ， 表 示 用 户 未 单 击 “登录 ”按钮 。 
(2) msg_ id， 表示 用 户 注册 结果 ， 其 可 选 值 如 下 : 
201， 表 示 用 户 已 注册 成 功 。 

可 选 参数 301， 表 示 用 户 类 和 姓名 类 参数 不 符合 要 求 。 


模型 
视图 


302， 表 示 密 码 设置 不 符合 要 求 。 
303， 表 示 未 上 传 相 
304， 表 示 相 片上 传 失败 。 

305， 表 示 指 定 的 用 户 名 已 经 存在 。 
401， 表 示 用 户 注册 失败 
app/controllers/main/register.php 
app/models/main/user.php 


亚 


app/Views/register.php 


main/logout 


app/controllers/main/logout.php 
无 


逐条 添加 收入 /消费 记录 


main/add/record_ type/action/msg ld 


record_type， 表 示 欲 添加 的 记录 类 型 ， 其 可 选 值 如 下 : 
income， 表 示 收 入 记录 。 
expense， 表 示 消 费 记录 


(1) action， 表 示 用 户 是 否 已 经 单 击 “ 添 加 收入 记录 ”或 “添加 消费 
记录 ”按钮 ， 其 可 选 值 如 下 : 

do， 表 示 用 户 已 单 击 “ 添 加 收入 记录 ”或 “添加 消费 记录 ”按钮 。 
(NULL) 

(2) msg id， 表示 添加 数据 记录 的 结果 ， 其 可 选 值 如 下 : 

201， 表 示 数 据 记 录 添 加 成 功 。 

301， 表 示 数 据 记 录 描 述 不 符合 要 求 。 

302， 表 示 日 期 格式 不 正确 。 

303， 表 示 所 涉 金 额 格式 不 正确 。 

401， 表 示 添 加 数据 记录 的 操作 失败 


控制 器 


app/controllers/main/add.php 
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项 目 数 据 

模型 app/models/incomes.php 和 app/models/expenses.php 

视图 app/views/add.php 

批量 添加 收入 /消费 记录 

路 由 main/b add/action/msg id 
(1) actionp， 表 示 用 户 是 否 已 经 单 击 “ 添 加 收入 记录 ”或 “添加 消费 
记录 ”按钮 ， 其 可 选 值 如 下 : 
do， 表 示 用 户 已 单 击 “ 添 加 收入 记录 ”或 “添加 消费 记录 ”按钮 。 
(NULL) 

可 选 参数 (2) msg_ id， 表示 批量 添加 数据 记录 的 结果 ， 其 可 选 值 如 下 : 


201， 表 示 数 据 记 录 批 量 添加 成 功 。 
301， 表 示 未 指定 账 务 文件 。 

302， 表 示 上 传 的 账 务 文件 格式 错误 。 
401， 表 示 批 量 添加 数据 记录 的 操作 失败 


控制 器 

模型 

视图 

查看 收入 /消费 记录 列表 
路 由 

必 选 参数 


可 选 参数 


app/controllers/main/b_add.php 
app/models/incomes.php 和 app/models/expenses.php 
app/views/b_add.php 


main/list/start_year/start_month/start_day/order_ type 

无 

(1) start_year/start_month/start_day， 表 示 记 录 的 开始 时 间 。 

(2) order type， 表 示 记 录 排 序 方式 ， 其 可 选 值 如 下 : 

asc， 表 示 按 记录 入 库 的 顺序 从 前 往 ， 最 新 记录 排 在 最 后 。 
desc， 表 示 按 记录 入 库 的 顺序 从 后 往 前 排序 ， 最 新 记录 排 在 最 前 


控制 器 

模型 

视图 

修改 指定 的 收入 /消费 记录 
路 由 


app/controllers/main/list.php 
app/models/incomes.php 和 app/models/expenses.php 
app/views/listphp 


main/modify/record_type/record_id/action/msg id 
(1) record type， 表 示 欲 修改 的 记录 类 型 ， 其 可 选 值 如 下 : 
incomes， 表 示 收 入 记录 。 


必 选 参数 expenses， 表 示 消 费 记录 。 
(2) record id， 表示 和 欲 修改 的 记录 编号 
(1) action， 表 示 用 户 是 和 否 已 经 单 击 “ 添 加 收入 记录 ”或 “添加 消费 
记录 ”按钮 ， 其 可 选 值 如 下 : 
do， 表 示 用 户 已 单 击 “添加 收入 记录 ”或 “添加 消费 记录 ”按钮 。 
人 (NULL) 
本 过 才 笋 (2) msg_ id， 返回 修改 结果 的 编号 ， 其 可 选 值 如 下 : 
201， 表 示 指 定 记 录 修改 成 功 。 
301， 表 示 用 户 输入 的 数据 不 符合 要 求 ， 提 示 用 户 重新 输入 。 
401， 表 示 修 改 操作 失败 
控制 器 app/controllers/main/modify.php 
模型 app/models/incomes.php 和 app/models/expenses.php 
视图 无 
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项 目 数 据 
删除 指定 的 收入 /消费 记录 
路 由 main/delete/record type/record id/action/msg id 
(1) record type， 表 示 欲 修改 的 记录 类 型 ， 其 可 选 值 如 下 : 
incomes， 表 示 收 入 记录 。 
必 园 参数 expenses， 表 示 消 费 记 录 。 
(2) record id， 表 示 欲 修改 的 记录 编号 
(1) action， 表 示 用 户 是 否 已 经 单 击 “ 添 加 收入 记录 ”或 “添加 消费 
记录 ”按钮 ， 其 可 选 值 如 下 : 
do， 表 示 用 户 已 单 击 “ 添 加 收入 记录 ”或 “添加 消费 记录 ”按钮 。 
(NULL) 
风光 和 (2) msg_ id， 返回 修改 结果 的 编号 ， 其 可 选 值 如 下 ; 
201， 表 示 指 定 记录 修改 成 功 。 
301， 表 示 用 户 输入 的 数据 不 符合 要 求 ， 提 示 用 户 重新 输入 。 
401， 表 示 修 改 操 作 失 败 
控制 器 app/controllers/main/delete.php 
模型 app/models/incomes.php 和 app/models/expenses.php 
视图 于 


在 向 框架 目录 中 添加 了 规划 的 各 种 控制 器 、 模 型 和 视图 以 及 使 用 Bootstrap UI 框架 所 
需 的 各 种 文件 之 后 ， 框 架 目 录 结 构 如 下 : 


1------chapter13 
1 一 一 apP 
|------controllers 
1 一 ----main 
1------add .php 
1------b add.php 
1------dashboard.php 
1------delete.php 
1------list.php 
1------login.php 
1------logout .php 
1-----modify.php 
1------register.php 


1------models 
|------expense .php 
1------income .php 
1-—----user .php 

Views 

1------add .php 
1-—----b add.php 
1-—----dashboard.php 
|-—----index.php 
1—--—list.php 
|-—----login.php 
1 一 一 -Logout .php 


二 这 二 
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1-—---register.php 


-C55 
------bootstrap.css //Bootstrap UI 框架 的 样式 表 
------bootstrap.min.css // 压 缩 后 的 Bootstrap UI 框架 样式 表 
------font-awesome.css //FontAwesome 图 标 字体 样式 表 
-一 -font-awesome-min-css  // 压 缩 后 的 FontAwesome 图 标 字 体 样式 表 
------morris.css //Morris 图 表 插 件 样 式 表 
------: Overwrite.css // 用 来 改写 Bootstrap UI 框架 样式 的 样式 表 

——=fonts 


------fontawesome-webfont .eot 

—— fontawesome-webfont .svg 

-一 fontawesome-webfont .tt 上 //FontAwesome 图 标 字 体 样 式 表 

------fontawesome-webfont .woff 

----- -glyphicons-halflings-regular.eot 

--- ‘glyphicons-halflings-regular.svg 

------glyphicons-halflings-regular.ttf //Boostrap UI 框架 引用 的 图 标 字 
体 文件 

------glyphicons-halflings-regular.woff 


1------image 


[一 js 


i avatar // 用 于 存放 用 户 上 传 的 头像 
------] bootstrap.js //Bootstrap UI 框架 用 Javascript 脚本 文件 


------bootstrap.min.js// 压 缩 后 的 Bootstrap UI 框架 用 JavaScript 脚本 文件 
------jquery-2.0.3.min.js//Bootstrap UI 框架 依赖 的 JavaSscript 库 


----- -morris.js //Morris 图 表 插件 JavaScript 脚本 文件 
3 -morris.min.js // 上 压缩 后 的 Morris 图 表 插 件 JavaScript 脚本 文件 
-1 upload // 用 于 存放 用 户 使 用 批量 导入 账 务 功能 时 上 传 的 账 务 文件 
I-———.htaccess 
|------index .php 
|------kissmvc.php 


|------kissmve_core.php 


到 这 里 ， 我 们 就 完成 了 数据 规划 。 在 下 面 的 编程 过 程 中 , 一 定 要 严格 遵守 规划 的 数据 ， 
以 便 脚 本 的 规模 可 控 、 脚 本 的 实现 可 管理 。 


13.3.2 用 户 登 录 与 验证 


按照 数据 规划 ， 本 页 面 主要 用 于 实现 用 户 登录 系统 时 对 用 户 身份 的 验证 工作 。 涉 及 到 
的 控制 器 、 模 型 和 视图 如 表 13-4 所 示 。 


Imain/login/action/msg id 


表 13-4 登录 页 面 使 用 的 控制 器 、 模 型 和 视图 


无 


vs 


(1) action ， 表 示 用 户 是 否 单 击 “ 登 录 ” 按 钮 ， 其 可 选 值 如 下 : 


do， 表 示 用 户 已 单 击 “ 登 录 ” 按 钮 。 


(CNULL) ， 表 示 用 户 未 单 击 “ 登 录 ” 按 钮 。 

(2) msg_ id， 表示 用 户 登 录 结 果 ， 其 可 选 值 如 下 : 
301， 表 示 用 户 输入 的 信息 不 符合 要 求 。 

401， 表 示 数 据 库 连接 失败 ， 无 法 登录 
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控制 器 app/controllers/main/login php 
模型 app/models/user.php 


视图 app/views/login.php 


现在 , 我们 需要 在 KISSMVC 框架 的 应 用 目录 下 的 控制 器 文件 夹 main 子 文件 夹 中 创建 
一 个 名 为 login.php 的 文件 ， 该 文件 主要 负责 用 户 身 份 验证 的 业务 逻辑 部 分 。 

【 例 13.1】 用 户 登录 页 面 的 控制 器 (app/controllers/main/login.php)。 

<?php 


function login($action="'',$msgId="'') { 


// 创 建 用 户 登录 页 面 的 视图 对 象 ， 并 引用 相应 的 视图 文件 
Sview = new View (APP PATH.'views/login.php'); 


// 定 义 一 些 必要 的 变量 

$msg=array ('alert-class'=>'',"'subject'=>'', 'body'=>"'"'); 
$username = 'username'; 

Spassword=' 7 


// 判 断 页 面 URL 中 是 否 包含 消息 码 ， 如 果 有 ， 则 将 消息 码 转换 成 消息 内 容 
if ($msgId){ 
Switch ($msgId){ 
case, "30."s 


Smsg["alert-class'] = "alert-warning'7 
$msg['subject'] = 'Warning'; 
$msg['body'] = 'Both the username and password must be 
specified.'; 
break; 

case '401°': 
Smsg['"alert-class'] = "alert-danger'7 
Smsg['subject'] = 'Error'; 
$msg['body'] = 'No match has been found. Check your 
inputs, please.'; 
break; 


} 
// 若 URL 中 没有 消息 码 ， 继 续 判断 URL 是 否 包含 动作 参数 ， 如 果 有 上 且 为 "qo"， 开 始 处 
理 用 户 提交 数据 
} elseif($action == "do') { 
// 获 取 用 户 通过 表单 提交 的 数据 
$username $ POST['username']; 
$password $ POST['password']; 


// 若 用 户 提交 的 数据 为 空 ， 则 通过 在 URL 中 指定 消息 码 输出 反馈 信息 


if(!Susername || !S$password) { 
header ('location:"' .WEB FOLDER.'main/login/do/301'); 
} else { 


// 若 用 户 提交 的 数据 不 为 室 ， 则 创建 一 个 用 户 对 象 

// 然 后 把 用 户 提交 的 数据 做 为 条 件 ， 通 过 创建 的 用 户 对 象 查找 数据 库 
// 并 尝试 获取 用 户 ID 和 全 名 

Suser = new User(); 

$user->retrieve one('username = ? AND password = ?2"， 
array ($username, S$password)); 

S$uid = $user->get ('uid'); 

$fullname = explode(' ', $user->get('fullname')); 


// 若 成 功 获取 到 用 户 TID， 表示 数据 库 中 存在 该 用 户 
// 因 此 ， 将 用 户 的 ID 和 命名 存 入 相应 的 SESSION 变量 中 ， 并 转向 控制 台 页 面 


Sr 
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if($uid) { 
$ SESSION['uid'] = S$uid; 
$ SESSION['firstname'] = $fullname[0]; 
header ('location:" .WEB FOLDER.'main/dashboard'); 
} else { 
// 若 获取 不 到 用 户 TID， 表示 数据 库 中 不 存在 匹配 用 户 
// 通 过 在 URL 中 指定 消息 码 输出 反馈 信息 
header ('location:"'.WEB FOLDER."'main/login/do/401°'); 


} 
// 将 用 户 信息 和 反馈 信息 输出 到 之 前 创建 的 视图 对 象 中 


Sview->set ('username', $username); 
Sview->set ('alert class', $msg['alert-class']); 
$view->set ('subject', $msg['subject"']); 
$view->set ('body', $msg['body']); 
$view->dump (); 
} 
2 


看 完 这 段 脚本 ， 大 家 应 该 会 发 现 ， 我 们 把 验证 用 户 身 份 的 业务 逻辑 全 部 写 在 了 控制 器 
中 ， 并 通过 控制 器 调用 了 USER 模型 和 app/views/login.php 视图 文件 。 通 过 控制 器 和 模型 
的 交互 ， 获 取 到 了 视图 文件 所 需要 的 所 有 变量 。 最 后 ， 使 用 视图 对 象 的 set 方法 将 获取 到 
的 所 有 变量 输出 到 视图 文件 中 。 

接 下 来 ， 我 们 在 应 用 目录 下 的 模型 文件 夹 中 创建 一 个 名 为 user.php 的 文件 。 该 文件 描 
述 了 数据 库 中 的 user 表 的 结构 。 脚 本 如 下 : 

【 例 13.2】 用 户 表 模型 (app/models/user.php) 。 


<?php 

class User extends Model { 

function User() { 
parent:: construct('uid','user','getdbh'); 
$this->rs['uid'] = "''; 
$this->rs['username'] 
S$this->rs['password'] 
$this->rs['fullname'] 
S$this->rs['gender'] 
$this->rs['avatar'] 
Sthis->rs['created dt' 


] 


} 
} 
> 
在 这 段 脚本 中 ,我们 定义 了 一 个 名 为 User 的 类 ， 并 在 User 类 中 定义 了 一 个 同名 方法 ; 
然后 通过 引用 其 父 类 的 _construct 方法 初始 化 了 User 类 ; 接着 , 把 user 数据 表 中 的 各 字段 
做 为 索引 添加 到 了 User 对 象 的 rs 属性 中 。 
最 后 ， 在 应 用 目录 下 的 views 文件 夹 中 创建 一 个 名 为 login.php 的 文件 ， 该 文件 描述 了 
用 户 登 录 页 面 的 布局 。 我 们 要 将 需要 动态 蔡 换 的 内 容 定义 成 变量 ， 这 样 一 来 ， 就 可 以 在 控 
制 器 中 对 这 些 变量 进行 设置 。 由 于 篇 幅 所 限 ， 视 图 文件 的 内 容 就 不 在 这 里 展示 了 。 大 家 一 
定 要 记 住 的 一 点 是 ， 一 定 不 要 在 视图 文件 中 定义 任何 业务 逻辑 ， 也 就 是 说 不 要 在 视图 文件 
中 做 任何 条 件 判断 .因为 在 视图 文件 中 做 条 件 判 断 会 严重 降低 视图 文件 的 可 维护 性 .图 13-8 
所 示 展 示 了 用 户 登 录 页 面 的 布局 。 
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PennyCounts 


'Cause every penny counts! 


图 13-8 用 户 登录 页 面 


13.3.3 ”用户 注册 


按照 数据 规划 ， 本 页 面 主要 用 于 实现 用 户 的 注册 。 涉 及 到 的 控制 器 、 模 型 和 视图 文件 
如 表 13-5 所 示 。 


表 13-5 ”注册 页 面 使 用 的 控制 器 、 模 型 和 视图 


路 由 main/register/action/msg id 

(1) action， 表 示 用 户 是 否 单 击 “ 注 册 ” 按 钮 ， 其 可 选 值 如 下 : 
do， 表 示 用 户 已 单 击 “ 注 册 ” 按 钮 。 

(NULL) ， 表 示 用 户 未 单 击 “ 登 录 ” 按 钮 。 

(2) msg_id， 表 示 用 户 注册 结果 ， 其 可 选 值 如 下 : 

201， 表 示 用 户 已 注册 成 功 。 

可 选 参数 301， 表 示 用 户 类 和 姓名 类 参数 不 符合 要 求 。 

302， 表 示 密 码 设置 不 符合 要 求 。 

303， 表 示 未 上 传 相片 。 

304， 表 示 相 片上 传 失败 。 


305， 表 示 指 定 的 用 户 名 已 经 存在 。 
401， 表 示 用 户 注册 失败 
控制 器 app/controllers/main/register.php 
模型 app/models/main/user.php 
视图 app/views/register php 


现在 ， 我 们 需要 在 应 用 目录 下 的 控制 器 文件 夹 下 的 main 子 文件 夹 中 创建 一 个 名 为 
register.php 的 文件 ， 它 定义 了 用 户 注册 的 业务 逻辑 ， 其 脚本 如 下 : 
【 例 13.3】 用 户 注册 页 面 的 控制 器 (app/controllers/main/register.php) 。 
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<?php 
function register($action="", $msgId=""){ 
$view = new View (APP PATH.'views/register.php'); 


$msg=array ("'alert-class'=>'"'alert-default','subject'=>'', 'body'=> 


呈 5 


if($msgId) { 
Switch (SmsgId) { 
case "201" : 


$msg['alert-class'] = "alert-success'7 
Smsg["subject'] = 'Success!'; 
$msg['body'] = "Congratulations! You've been one of us, 
finally!"; 
break; 

case "301": 
$msg["'alert-class'] = "alert-warning'"7 
$msg['subject'] = "Warning!'; 
$msg['body'] = 'The username, first name, and last name 
each 

must be a string of 3 to 24 characters.'; 

break; 

case '302°': 
$msg['alert-class'] = 'alert-warning'; 
Smsg['"subject'] = 'Warning!'; 
$msg['body'] = 'The password must be a string of 7 to 


15 characters, 
and ensure that you have enterred the same 
password twice.'; 


break; 

case "303 ' : 
Smsg["alert-class'] = "alert-warning'7 
$msg['subject'] = 'Warning!'; 
$msg['body'] = 'You have to upload a photo of yourself.'; 
break; 

case "304 ' : 
Smsg['"alert-class'] = "alert-warning'"7 
Smsg['subject'] = 'Warning!'; 
$msg['body'] = "The image fails to be uploaded. Please 
contact the administrator.'; 
break; 

case "305 ' : 
Smsg['"alert-class'] = "alert-warning'7 
$msg['subject'] = 'Warning!'; 
$msg['body'] = 'The username you specified already 
exists. Please enter another one.'; 
break; 


case '401°': 


$msg["'alert-class'] = ‘alert-error'; 
$msg['subject'] = "Error!'; 
$msg['body'] = 'Something goes wrong with our database! 


Please sign youself up some day in the near 
future!'; 
break; 
’ 
} elseif ($action == "do') { 


// 获 取 用 户 提交 的 数据 
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Susername = $ POST['username']; 
$firstname = $ POST['firstname']; 
$lastname = $_POST['lastname']; 
Spassword = $_POST['password']; 
$confirm = $ POST['confirm']; 
S$Sgender = $ POST['gender']; 


Suser = new User(); 
$user->retrieve one('username=?', array ($username)); 
Suid = $user->get ('uid'); 


// 验 证 用 户 输入 数据 

$fault id = 0; 

if(!auth strings (array ($username, $firstname, $lastname) ,2, 25)) { 
$fault id = 1; 

} elseif(!auth strings (array ($password, $confirm), 6, 16)) { 
$fault id = 2; 

} elseif(!auth image file($ FILES['photo'])) { 
$fault id = 3; 

} elseif($uid) { 
$fault id = 5; 

} 


// 若 用 户 输入 的 信息 有 误 ， 则 转向 相应 警告 页 面 ， 否 则 开始 上 传 文件 

if($fault id > 0){ 

header ('location:"' .WEB FOLDER. 'main/register/do/30'.S$ftault id) ; 
} else { 

wast 

Sphoto dest="\\img\\avatar\\".base64 encode ($username) .".". 

substr($ FILES['photo'] ['type'],6); 
move_ uploaded file($ FILES['photo']['tmp name'],ABSOLUTE WEB_ 
ROOT.$photo dest); 


// 判 断 已 上 传 的 文件 是 否 存在 ， 若 存在 ， 则 开始 写 入 数据 库 ， 否 则 转向 相应 警告 页 面 
if(file_exists (ABSOLUTE WEB ROOT.$photo dest)){ 
// 开 始 写 入 数据 库 
Suser->set('username'，S$Susername) 7 
Suser->set ('fullname', $firstname.' '.$lastname); 
$user->set ('gender', S$gender); 
$user->set ('password', S$password); 
$user->set ('avatar', S$photo dest); 
$user->set ('created dt',date('Y-m-d')); 
Suser->create(); 
$uid = $user->get ('uid'); 
if(Suid) { 
$ SESSION['uid'] = $uid; 
$ SESSION['firstname'] = $firstname; 
header ('location:"' .WEB FOLDER.'main/register/do/201°'); 
} else { 
header ('location:"'.WEB FOLDER."'main/register/do/401"'); 
} 
} else { 
header ('location:"'.WEB FOLDER."'main/register/do/304'); 
} 


中 


Sview->set('alert class', $msg['alert-class']); 
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$view->set ('subject', $msg['subject"']); 
Sview->set ('body', $msg['body']); 
Sview->dump (); 

> : 

在 上 面 这 段 脚本 中 ， 业 务 逻 辑 的 实现 与 用 户 登录 页 面 类 似 ， 都 是 先 判 断 是 否 存 在 消息 
码 。 若 存在 ， 则 将 消息 码 转 换 成 相应 的 消息 ， 然 后 输出 到 页 面 ， 若 不 存在 ， 则 继续 判断 用 
户 是 否 提交 了 表单 。 若 提交 了 ， 则 开始 获取 并 验证 用 户 数 据 。 

值得 注意 的 是 ， 我 们 在 处 理 用 户 上 传 的 照片 文件 时 ， 使 用 了 一 个 名 为 
ABSOLUTE WEB_ ROOT 的 常量 , 该 常量 存储 的 是 框架 根 目录 在 服务 器 上 的 绝对 路 径 。 大 
家 如 果 也 想 定义 一 些 常 量 ， 可 以 在 框架 根 目录 下 的 index.php 文件 中 添加 。 

另外 ， 我 们 在 验证 用 户 提交 的 数据 时 ， 使 用 了 三 个 自 定义 的 函数 。 如 果 大 家 需要 在 某 
个 控制 器 中 使 用 自 定义 的 函数 ， 可 以 直接 在 该 控制 器 页 面 的 尾部 (“?>”) 前 添加 。 我 们 
在 用 户 注册 页 面 上 定义 的 三 个 用 于 验证 用 户 提交 的 数据 的 函数 如 下 : 

【 例 13.4】 用 户 注册 页 面 的 控制 器 中 的 自 定义 函数 (app/controllers/main/register.php)。 


function auth strings($strings, $minlen, $maxlen) { 
$false count = 0; 
foreach ($strings as $str) { 
if(strlen($str) >= Smaxlen || strlen($str) <= $minlen) 
$false count++; 


if($false count){ 
return FALSE; 
} else { 
return TRUE; 
} 
1 


function auth passwords ($password, S$confirm, $minlen, $maxlen){ 
if(auth strings (array($password, $confirm), $minlen, $maxlen) && 


$password == $confirm){ 
return TRUE; 
} else { 


return FALSE; 
站 
} 


function auth image file($file){ 
if($file['name'] && $file['tmp name'] && preg match('/image/', 
$file['type'])){ 
return TRUE; 
} else { 
return FALSE; 
} 
} 


上 面 这 段 脚 本 中 定义 了 函数 实现 的 功能 都 很 简单 ， 只 是 可 以 保证 在 向 数据 库 中 写 入 用 
户 输出 的 数据 不 会 出 现 错误 。 大 家 在 实际 的 开发 中 ， 应 该 加 强 对 用 户 输出 数据 的 验证 。 

本 页 面 与 用 户 登 录 页 面 一 样 ， 都 引用 了 User 模型 。 关 于 User 模型 的 脚本 ， 大 家 可 以 
参考 例 13.2 的 内 容 。 图 13-9 展示 了 用 户 注册 页 面 的 布局 。 
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图 13-9 用 户 注册 页 面 


13.3.4 添加 收入 和 支出 记录 


按照 数据 规划 ， 本 页 面 主要 用 于 实现 向 数据 库 中 添加 收入 和 支出 记录 。 涉 及 到 的 控制 
器 、 模 型 和 视图 文件 如 表 13-6 所 示 。 


表 13-6 添加 收入 和 支出 记录 页 面 使 用 的 控制 器 、 模 型 和 视图 
路 由 Imain/add/record type/action/msg id 
TIecord _ type， 表示 欲 添 加 的 记录 类 型 ， 其 可 选 值 如 下 : 
必 选 参数 | income， 表 示 收 入 记录 。 
expense， 表 示 消 费 记录 
(1) action， 表 示 用 户 是 否 已 经 单 击 “ 添 加 收入 记录 ”或 “添加 消费 记录 ”按钮 ， 其 可 选 
值 如 下 : 
do， 表 示 用 户 已 单 击 “ 添 加 收入 记录 ”或 “添加 消费 记录 ”按钮 。 
(NULL) ， 表 示 该 参数 未 指定 ， 用 户 未 单 击 上 述 按钮 。 
(2) msg_id， 表 示 添 加 数据 记录 的 结果 ， 其 可 选 值 如 下 : 
201， 表 示 数 据 记 录 添 加 成 功 。 
301， 表 示 数 据 记 录 描 述 不 符合 要 求 。 
302， 表 示 日 期 格式 不 正确 。 
303， 表 示 所 涉 金额 格式 不 正确 。 
401， 表 示 添 加 数据 记录 的 操作 失败 
控制 器 app/controllers/main/add php 


模型 app/models/incomes.php 和 app/models/expenses.php 
视图 app/views/add.php 


现在 ， 我 们 需要 在 应 用 目录 下 的 控制 器 文件 夹 下 的 main 子 文件 夹 中 创建 一 个 名 为 


add.php 的 文件 ， 它 定义 了 用 户 向 数据 库 中 添加 收入 和 支出 记录 的 业务 逻辑 ， 其 脚本 如 下 : 
【 例 13.5】 添加 收入 和 支出 记录 页 面 的 控制 器 (app/controllers/main/addphp) 。 
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<?php 
function add($recordType, $action="", $msgId="") { 
// 判 断 用 户 是 和 否 已 经 登录 ， 未 登录 则 转向 登录 页 面 。 
if(!$ SESSION["uid"]) { 
header ('location:" .WEB FOLDER); 


} else { 
Sview = new View (APP PATH.'views/add.php'); 
Smsg = array("alert-class' => '', 'subject' => '', 'body' => ''); 
// 定 义 所 有 可 选 记录 类 别 
$category['income'] = array('SW' => "Salary & Wages', 


'BN' => 'Bonus', 

"DD' => "Dividend'， 

"RT' => "Rentals' 

MORE => 7OEhersry > 
$category['expense'] = array('GD' => "Garment & Dressing', 

'FD' => "Food & Vegetable '， 

"HR' => "Housing & Rentals', 

1"UT' => "Utilities & Others' 

'TP' => "Transportation') 7 


// 定 义 页 面 上 的 标语 和 相应 的 可 选 记录 类 别 
switch($recordType) { 
case "income ' : 
$slogan = 'Treasuring every penny You earn makes You a 
rich man..- "7 
$category = $category['income']; 
break; 
case "expense ' : 
$slogan = "Spend with control, being prosperous as you 
know..."; 
$category = $category['expense']; 
break; 


} 


// 若 URL 中 包含 smsgTd， 则 根据 SmsgId 的 值 来 判断 展示 的 内 容 
if(SmsgId) { 
Switch (SmsgId) { 
case "201' : 


$msg["'alert-class'] = "alert-success' 7 
$msg['subject'] = 'Success' 7 
$msg['body'] = "Succeeded in adding the '. 
$recordType.' record.'; 
break; 

case "301": 
Smsg['alert-class'] = 'alert-warning'; 
Smsg['subject'] = 'Warning'7 


$msg['body'] = 'The description must be a string of 
5 to 128 characterss > 


break; 

case '302': 
$msg["'alert-class'] = 'alert-warning'; 
$msg['subject'] = 'Warning'; 
$msg['body'] = 'The date must be specified in the 
YYYY=MM=DD, formats zx 
break; 

case 303 
$msg["'alert-class'] = "alert-warning'"7 
$msg['subject'] = 'Warning'; 


$msg["'body'] = 'The amount must be a float number with 
two bits behind the decimal dot.'; 
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break; 
case '401°': 
Smsg["alert-class'"] 


"alert-danger'; 


$msg['subject'] = "Error'; 

$msg['body'] = "Failed to add the '‘'.$recordType."' 
record.'; 

break; 


} 
// 若 URL 中 不 包含 SimsgId, 却 包含 Saction， 则 开始 获取 并 验证 用 户 输入 的 
信息 
} elseif($action == "do'){ 
$description = $ POST['description']; 
$date = $ POsT['date']; 
Samount = $ POST['amount']7 
$selected category = $ POST['category']; 


// 验 证 用 户 输入 的 信息 

$fault = 0; 

if(strlen($description) > 128 || strlen($description) < 5) { 
$fault = 1; 

} elseif(!'auth date($date)) { 
$fault = 2; 

} elseif(!auth float ($amount)){ 
$fault = 3; 

} 


// 若 验证 未 通过 ， 展 示 相 应 告警 信息 。 若 验证 通过 ， 则 开始 将 用 户 输入 的 信息 
写 入 数据 库 
if($fault){ 
header ('location:"'.WEB FOLDER.'main/add/'.$recordType. '/do 
/30" .$fault)s 
} else { 
Switch ($recordType) { 
case ‘'income': 
$income = new Income(); 
$income->set ('description', $description); 
$income->set ('amount', S$amount); 
$income->set ('add dt', $date); 
$income->set ('mod dt',S$date); 
$income->set ('category', $selected category); 
$income->set ('uid',$ SESSION['uid']); 
$income->create(); 
$incid = $income->get ('incid'); 
if($incid) { 
header ('location:"' .WEB FOLDER.'main/add/ 
income /do/201°'); 
} else { 
header ('location:"' .WEB FOLDER.'main/add/ 
income/do/401°'); 
} 


break; 


case "expense ' : 
S$expense = new Expense(); 
Sexpense->set ('description', $description); 
$expense-—>set ('amount', $amount); 
Sexpense->set ('add dt', $date); 
Sexpense->set ('mod dt',S$date); 
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Sexpense->set ('category', $selected 
category); 
Sexpense->set ('uid',$ SESSION['uid']); 
Sexpense->create (); 
$expid = $expense->get ('expid'); 
if($expid) { 
header ('location:"' .WEB FOLDER."'main/add 
/expense/do/201°'); 
} else { 
header ('location:"'.WEB FOLDER.'main/add 
/expense/do/401"'); 
上 


break; 


上 


S$Sview->set ('recordType'，S$recordType) 
S$Sview->set ('slogan', $slogan); 

Sview->set ('alert class', $msg['alert-class']); 
S$Sview->set ('subject', $msg['subject"']); 
S$Sview->set ('body', $msg['body']); 

S$Sview->set ('category', $category); 
Sview->dump (); 


> 


在 上 面 这 段 脚本 中 ， 我 们 为 了 验证 用 户 提交 的 数据 ， 也 定义 了 两 个 自 定义 函数 ， 其 脚 
本 如 下 : 
function auth date($date str){ 
Sregeexp 0 /oe a NANar) WOoRL3Srauto2 (0 ll 
Pi2INal3 [00] 
(((1[6-9]1[2-9]\d) \df2})-(0?[13456789]11[012])-(02[1-9]1 [12]\dl30))1 
(ULG OM NON :02 (0 SN Do ure ol2 = 
\d) (0[48] | [2468] [048] | 
[13579] [26])1((161[2468] [048] | [3579] [26] ) 00))-0?22-29-))$/"; 
if(preg match ($reg exp, S$date str)) { 
return TRUE; 
} else { 
return FALSE; 
} 
} 


function auth float ($float) { 
Sregp exp = NAL tN ES 
if (preg match ($reg exp, $float)){ 
return TRUE; 
} else { 
return FALSE; 
} 
| 


这 两 个 验证 函数 都 使 用 了 正则 表达 式 。 其 中 用 于 验证 日 期 格式 的 正则 表达 式 比较 复 
杂 ， 大 家 可 以 保存 下 来 ， 以 便 后 续 使 用 。 图 13-10 和 图 13-11 分 别 展示 了 添加 收入 和 支出 
记录 页 面 的 布局 。 
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Add an Income Record 


Treasuring every pennyyou earn makes you a rich man... 


图 13-10 添加 收入 记录 


Md mepemae teend 下 
和 了 了 CD wwwareawallteay 
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Add an Expense Record 


Spend with control, being prosperous as you know... 


Description 


Date 


Category 


图 13-11 添加 支出 记录 


在 向 数据 库 中 添加 收入 和 支出 记录 的 过 程 中 ， 我 们 还 引用 了 Income 类 和 Expense 类 。 
为 了 定义 这 两 个 类 ， 需 要 在 应 用 目录 下 的 模型 文件 夹 中 创建 两 个 名 为 income.php 和 
expense.php 的 文件 ， 其 脚本 如 例 13.6 和 例 13.7 所 示 。 

【 例 13.6】 收入 记录 表 模 型 (app/models/income.php) 。 

<?php 


class Income extends Model { 
function Income() { 


parent:: construct('incid','income','getdbh'); 
th le hI eh Ke be 

sthis => rsl"desceription"]l = 

sthis =—> rsl"'amount"] 三 77 


ws 
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Sthis -> rs['add dt'] 
Sthis -> rs['mod dt'] 
Sthis -> rs['category'] = 
SEhie EGG = 


} 
> 


【 例 13.7】 支出 记录 表 模 型 (app/models/expense.php) 。 


<?php 
class Expense extends Model { 
function Expense() { 
parent:: construct('expid','expense','getdbh'); 
Sthis -> rs['expid'] = "7 
Sthis -> rs['description'] = "7 
Sthis -> rs['amount'] = "''; 
Sthis -> rs['add dt'] = 
Sthis -> rs['mod dt'] = 
$this -> rs['category'] 
sthis. => Ts nid™ ll = "sy 


} 
> 


至 此 ， 我 们 这 个 记 账 工 具 中 使 用 到 的 所 有 模型 就 定义 完毕 了 。 
13.3.5 ”批量 添加 收入 和 支出 记录 


按照 数据 规划 ， 本 页 面 主要 用 于 实现 向 数据 库 中 批量 添加 收入 和 支出 记录 。 涉 及 到 的 
控制 器 、 模 型 和 视图 文件 如 表 13-7 所 示 。 


表 13-7 添加 收入 和 支出 记录 页 面 使 用 的 控制 器 、 模 型 和 视图 


路 由 main/b add/action/msg id 

必 选 参数 无 
(1) action， 表 示 用 户 是 否 已 经 单 击 “ 添 加 收入 记录 ”或 “添加 消费 记录 ”按钮 ， 
其 可 选 值 如 下 : 


do， 表 示 用 户 已 单 击 “ 添 加 收入 记录 ”或 “添加 消费 记录 ”按钮 。 
(NULL) ， 表 示 该 参数 未 指定 ， 用 户 未 单 击 上 述 按钮 。 

可 选 参数 (2) msg_id， 表 示 批 量 添加 数据 记录 的 结果 ， 其 可 选 值 如 下 : 
201， 表 示 数 据 记录 批量 添加 成 功 。 

301， 表 示 未 指定 账 务 文件 。 

302， 表 示 上 传 的 账 务 文件 格式 错误 。 

401， 表 示 批 量 添加 数据 记录 的 操作 失败 


控制 器 app/controllers/main/b add.php 
模型 app/models/incomes.php 和 app/models/expenses.php 
视图 app/models/views/b_add.php 
现在 ， 我们 需要 在 应 用 目录 下 的 控制 器 文件 夹 下 的 main 子 文件 夹 中 创建 一 个 名 为 
b_add.php 的 文件 ， 它 定义 了 用 户 向 数据 库 中 添加 收入 和 支出 记录 的 业务 逻辑 ， 其 脚本 
如 下 : 


【 例 13.8】 批量 添加 收入 和 支出 记录 页 面 的 控制 器 (app/controllers/main/b add.php) 。 


.364 


第 13 章 PHP 与 MVC 


<?php 
function b add($action="", $msgId=""){ 

Sview = new View (APP PATH.'views/b add.php'); 

Smsg = array('alert-class' => '', 'subject' => '', 'body' => ''); 

if(isset($ SESSION['uid'])){ 

if($msgId){ 
switch($msgId) { 
Sasey 200"s 
Smsg["alert-class'] = "alert-success' 7 
Smsg['"subject'] = "Success! 7 
Smsg["bodqy'] = "Congratulations! All records in the 
template file 
has been added to the database!"; 


break; 

case 301"s 
Smsg["alert-class'] = "alert-warning'7 
$msg['subject'] = 'Warning! 7 
$msg['body'] = 'No template file has been selected. 

You must select a file before we can 
contine.'; 

break; 

casel "302 
Smsg["alert-class'] = "alert-warning'7 
$msg['subject'] = 'Warning! 7 
$msg['body'] = 'The template file you have uploaded 
is not in the correct format.'; 
break; 

case "401' : 
Smsg["alert-class'] = "alert-danger'7 
$msg['subject'] = 'Error!'; 
$msg['body'] = 'Something goes wrong with our 
database! Please try again later!'; 
break; 

+ 
} elseif ($action == "do"){ 


// 若 用 户 提交 了 表单 ， 则 获取 用 户 提交 的 内 容 


$file = $ FILES['template']; 
Sname = $file['name']; 

Stemp = $file['tmp name']; 
$type = $file['type']; 


// 检 测 用 户 是 否 选择 了 一 个 文本 文件 
if($name && $temp && $type && preg match('/text/', $type)){ 


// 车 用 户 选择 了 一 个 文本 文件 ， 则 开始 上 传 

$dest = "\\upload\\".base64 encode($ SESSION['firstname 
i 

move uploaded file($temp, ABSOLUTE WEB ROOT.S$dest); 


// 检 测 文件 上 传 是 否 成 功 
if (file exists (ABSOLUTE WEB ROOT.S$dest)){ 


// 若 成 功 ， 则 开始 读 取 文件 内 容 ， 并 将 收入 记录 和 消费 记录 分 别 
// 以 数组 的 形式 存 入 两 个 变量 中 
Srecords = readTemp (ABSOLUTE WEB ROOT.S$dest); 
if(!is array (S$records)) 
header ('location:"' .WEB FOLDER.'main\b add\do 
N302")3 


$incomes = S$records['incomes']; 
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$expenses = S$records['expenses']; 


// 检 查收 入 记录 是 否 为 空 
if($incomes){ 
// 若 收入 记录 不 为 空 ， 则 开始 逐条 将 收入 记录 写 入 数据 库 
SIncErrCount = 0 
foreach ($incomes as S$record) { 
$income = new Income(); 
$income->set ("description", $record 
['description']); 
$income->set ("amount", $record['amount']); 
$income->set ("add dt",$record['added dt']); 
$income->set ("mod dt",$record['added dt"']); 
$income->set ("category", $record['category']); 
$income->set ("uid",$ SESSION['uid']); 
Sincome->create (); 
$incid = $income->get ('incid'); 
if(!$incid){ 
$incErrCount++; 


» 
} 
// 检 查 消费 记录 是 否 为 空 


if ($expenses){ 
// 若 消费 记录 不 为 室 ， 则 开始 逐条 将 消费 记录 写 入 数据 库 
$expErrCount = 0; 
foreach ($expenses as S$record) { 
Sexpense = new Expense(); 
Sexpense->set ("description", $record 
['description']); 
$expense->set ("amount", $record['amount']); 
$expense->set ("add dt",$record 
["added dt"']); 
$expense->set ("mod dt",$record 
['added dt"']); 
Sexpense->set ("category", $record 
['category']); 
$expense->set ("uid",$ SESSION['uid']); 
$expense->create (); 
Sexpid = $expense->get ('expid'); 
if(!$expid){ 
SexpErrCount++; 
) 
} 
} 
if(!$incErrCount && !$expErrCount){ 
// 若 在 写 入 数据 库 的 过 程 中 没有 出 现 错误 ， 则 表示 操作 成 功 ， 
返回 201 页 面 
header ('location:' .WEB FOLDER.'main/b add 
/do/201'); 
} else { 
header ('location:'.WEB FOLDER."'main/b add 
/do/401'); 


} 
} else { 

header ('location:"'.WEB FOLDER.'"'main/b add/do/301°'); 
} 
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} else { 
header ('location:' -WEB FOLDER) 


} 


Sview->set("'alert class', $msg['"'alert-class']); 
S$Sview->set('subject', $msg['subject"']); 
S$Sview->set ('body', $msg['body']); 
Sview->dump (); 


} 
> 


在 上 面 的 脚本 中 ， 我 们 也 使 用 了 一 个 自 定义 的 函数 ， 用 来 读 取 用 户 上 传 的 账 务 文件 。 
账 务 文件 的 模板 ， 大 家 可 以 在 框架 根 目 录 的 upload 文件 夹 下 找到 ， 其 内 容 如 下 : 


{EIDE}How do you describe the spending? 

{EIAM}How must have you paid? 

{EIAD}When did you pay for the things you have just described? 
{EICA}How do you categorize the spending? 


{IIDE}How do you describe the expense? 

{IIAM}How must have you earned? 

{EIAD}When did you earn the money you have just described? 
{IICA}How do you categorize the money you have just described? 


{EIDE} 

{EIAM} 

{EIAD} 

{EICA} 

大 家 可 以 通过 回答 模板 文件 中 的 问题 来 填写 模板 。 如 果 模 板 文件 中 的 条 目 数据 不 够 ， 
可 以 通过 复制 的 方法 进行 扩展 。 为 了 读 取 用 户 填 写 好 的 账 务 文 件 ， 定 义 了 一 个 名 为 
readTemp 的 函数 。 相 较 于 第 10 章 的 实战 练习 里 定义 的 那个 readTempO 函 数 ， 我 们 做 了 
些 改进 ， 让 其 能 够 将 读 取 到 的 数据 按照 既定 的 格式 存 入 一 个 数组 中 。 这 样 一 来 ， 便 可 以 通 
过 遍历 这 个 数组 来 完成 向 数据 库 中 添加 数据 记录 的 任务 。 脚 本 的 运行 效率 相 比 之 前 更 高 。 
该 函数 的 脚本 如 下 : 


function readTemp ($file){ 
$fh = fopen ($file, 'r'); 
Srecords = ''; 
$1 = O07 
while(!feof ($fh)){ 


$lines[$i] = fgets ($fh); 
SI 


} 
fclose ($fh); 


mb internal encoding('UTF-8°'); 
if(count ($lines)®%5 == 0){ 


Sa=07 

Sb=07 

for ($i=0; $i < count ($lines); $i=$i+5) { 
if(mb substr($lines[$i+1],1,1) == "I'){ 


Srecords [ "incomes '] [$al['description'] = trim(mb substr 
($lines[$i+1], 6)); 

$records['incomes'] [Sa] ["amount"] = trim(mb substr ($lines 
[$i+2], 6)); 
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S$records['incomes'] [Sa]l ['added dt'] = trim(mb substr 
($lines[$i+3], 6)); 
S$records['incomes'] [$al] ['category'] 
($lines[$i+4], 6)); 
Satt+; 
} else { 
Srecords [ "expenses'] [$b]['description'] = trim(mb substr 
($lines[$i+1], 6)); 
$records['expenses'] [$b]['amount'] = trim(mb substr 
($lines[$i+2], 6)); 
$records['expenses'] [$b] ['added dt'] 
($lines[$i+3], 6)); 
$records['expenses'] [$b]['category'] = trim(mb substr 
($lines[$i+4], 6)); 
$b++; 


trim(mb substr 


1 


trim(mb substr 


上 


return $records; 


图 13-12 展示 了 批量 添加 收入 和 支出 记录 的 页 面 布局 。 


Add Records ina Batch 


Take every penny you earn serious and spend them with control.. 


RR | 


图 13-12 ”批量 添加 收入 和 支出 记录 


13.3.6 ”查看 数据 记录 


按照 数据 规划 ， 本 页 面 主要 用 于 实现 从 数据 库 中 读 取 已 添加 收入 和 支出 记录 并 以 列表 
的 形式 展现 这 些 数据 。 涉 及 到 的 控制 器 、 模 型 和 视图 文件 如 表 13-8 所 示 。 
表 13-8 ”查看 数据 记录 页 面 使 用 的 控制 器 、 模 型 和 视图 


路 由 main/list/start_year/start month/start_day/order type 
必 选 参数 无 
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(1) start_year/start_ month/start_ day， 表 示 记 录 的 开始 时 间 。 

(2) order type， 表 示 记 录 排 序 方式 ， 其 可 选 值 如 下 : 

asc， 表 示 按 记录 入 库 的 顺序 从 前 往 后 排序 ， 最 新 记录 排 在 最 后 。 
desc， 表 示 按 记录 入 库 的 顺序 从 后 往 前 排序 ， 最 新 记录 排 在 最 前 


控制 器 app/controllers/main/list.php 


模型 | app/models/incomes.php 和 app/models/expenses.php 


视图 app/views/list.php 
现在 ， 我们 需要 在 应 用 目录 下 的 控制 器 文件 夹 下 的 main 子 文件 夹 中 创建 一 个 名 为 
list.php 的 文件 , 它 定义 了 从 数据 库 中 读 取 已 添加 的 收入 和 支出 记录 的 业务 逻辑 , 其 脚本 如 下 : 
【 例 13.9】 查看 数据 记录 页 面 的 控制 器 (app/controllers/main/list.php)〉。 


<?php 


function list($startYear="",$startMonth="", $startDay="",$sortBy 


="desc") 


{ 


Sview = new View (APP PATH.'views/list.php'); 
Srecords = array(array('id' => '', 


Ds 


'type' => '', 
"description' => '', 
"income' => ' 
"expense' => '', 
"category’ => "人 
"mod date' => '" 


if(!$ SESSION['"uid'"']){ 


header ('location:' .WEB FOLDER); 


} else { 


// 获 取 用 户 信息 

Suser = new User() ; 
$user->retrieve($ SESSION['uid']); 
$fullname = $user->get ('fullname'); 
Savatar = $user->get ('avatar'); 


if($startYear && $startMonth && $startDay){ 
$date = $startYear.'-'.$startMonth.'-'.$startDay; 
$startdate = date('M. n, Y',strtotime ($date)); 


// 读 取 该 用 户 的 收入 记录 

Sincome = new Income(); 

$rs = $income->retrieve many('uid = ? AND 

UNIX TIMESTAMP (mod dt) >= UNIX TIMESTAMP(?)', array($ SESSION 

['uid'], $date)); 

Si = 0; 

$totalIncome = 0; 

foreach ($rs as $index => $object) { 
S$records[$i]['id'] = $object->rs['incid']; 
S$records[$i]['type'] = 'income'; 
S$records[$i]['description'] = 
S$object->rs['description']; 
S$records[$i]['income'] = $object->rs['amount']; 
S$records[$i]['expense'] = "7 
S$records[$i]['category'] = getCategory ($object-—>rs 
['category']); 
Srecords [Si] ["mod date'] = $object->rs['mod dt']; 
SiT++7 
StotalIncome = StotalIncome + S$object->rs['amount']; 
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2 


在 上 面 这 段 脚 本 中 ， 大 家 可 以 仔细 体会 一 下 我 们 是 怎样 揉 合 收入 和 消费 记录 ， 并 将 它 


1 


// 读 取 该 用 户 的 消费 记录 


Sexpense = new Expense () 7 

Srs = $expense->retrieve many('uid = ? AND 

UNIX TIMESTAMP (mod dt) >= UNIX TIMESTAMP(?)', array($ SESSION 

["uid'], $date)); 

StotalExpense = 0; 

foreach ($rs as $index => $object) { 
S$records[$i]['id'] = $object->rs['expid']; 
S$records[$i]['type'] = 'expense'; 
S$records[$i]['description'] = 
S$object->rs['description']; 


$records[$i]['income'] = "'"'; 
Srecords [$i] ["expense'] = $object->rs['amount']; 
S$records[$i]['category'] = getCategory (Sobject->Ts 
['category']); 

S$records[$i]['mod date'] = $object->rs['mod dt']; 
EE 


$totalExpense = $totalExpense + $object->rs['amount']; 


} 


if(($totalIncome - S$totalExpense) <= 0) { 
SbalanceClass = 'text-danger'; 

} else { 
$balanceClass = 'text-success'; 


1 


// 揉 合 收入 和 消费 记录 ， 并 按 指 定 的 时 间 顺 序 排序 
foreach ($records as $index => S$arr) { 
$dtArr[] = strtotime ($arr['mod date']); 


} 

if ($sortBy == 'desc'){ 
arsort ($dtArr); 

} else { 
asort ($dtArr); 

1 


foreach ($dtArr as $idx => $dt) { 
$sortedRecords[] = $records[$idx]; 
} 


Sview->set ('fullname', $fullname); 
Sview->set ('avatar', S$avatar); 

Sview->set ('records', $sortedRecords); 
$view->set ('totalIncome', S$totalIncome); 
Sview->set ('totalExpense', S$totalExpense); 
Sview->set('startdate', $startdate); 
Sview->set ('balanceClass', $balanceClass); 
$view->dump (); 


们 按 指定 的 时 间 顺 序 排序 的 。 弄 懂 这 段 脚本 可 以 帮助 大 家 更 好 地 理解 数组 。 
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另外 ， 我 们 在 脚本 中 还 使 用 了 一 个 名 为 getCategory0 的 函数 ， 用 来 将 从 数据 记录 中 的 
category 字段 的 值 替 换 成 易于 理解 的 短语 。 其 脚本 如 下 : 


function getCategory ($string){ 
switch ($string) { 

case 'SW': 
$string = "Salary & Wages'; 
break; 

case 'BN': 
$string = 'Bonus'; 
break; 

case "DD' : 
$string = 'Dividend'; 
break; 

case 'RT': 
$string = 'Rentals'; 
break; 

case "OFT": 
$string = 'Others'; 
break; 

case "GD ' : 
$string = 'Garment & Dressing'; 
break; 

Casei ED: : 
$string = 'Food & Vegetable' 7 
break; 

case 'HR': 
$string = 'Housing & Rentals'; 
break; 

case “UTS: 
$string = "Utilities & Others'; 
break; 

case "TP' : 
$string = "Transportation'7 
break; 


} 
return $string; 


} 
图 13-13 展示 了 查看 数据 记录 页 面 的 布局 。 


PennyCounts 


View Reports 


Makes a list of what you have earned and spent. 


last Modified 。 Operations 


图 13-13 查看 数据 记录 
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13.3.7 ”控制 台 


按照 数据 规划 ， 本 页 面 主要 用 于 实现 从 数据 库 中 读 取 已 添加 收入 和 支出 记录 并 根据 这 
些 记 录 生 成 相应 的 统计 报表 。 在 这 个 页 面 中 ， 我 们 使 用 了 开源 的 jQuery 图 表 插 
件 一 一 Moris， 关 于 Morris 插件 的 使 用 方法 已 经 超越 了 本 书 的 范畴 ， 大 家 可 以 自行 参考 其 
官方 网 站 的 内 容 。 本 页 面 涉及 到 的 控制 器 、 模 型 和 视图 文件 如 表 13-9 所 示 。 


表 13-9 控制 台 页 面 使 用 的 控制 器 、 模 型 和 视图 


无 


无 


app/controllers/main/dashboard.php 


模型 app/models/incomes.php 和 app/models/expenses.php 

视图 app/views/dashboards.php 
由 于 这 个 页 面 没有 任何 参数 。 因 此 ， 业 务 逻 辑 会 比较 复杂 。 我 们 一 起 来 看 一 下 脚本 : 
<?php 


function dashboard() { 
Sview = new View (APP PATH.'views/dashboard.php'); 


if(!isset($ SESSION['uid'])){ 
header ('location:"' .WEB FOLDER); 

} else { 
// 获 取 用 户 信息 
Suser = new User() ; 
Suser->retrieve($ SESSION['uid']) > 
$fullname = $user->get('fullname'); 
Savatar = $user->get ('avatar'); 


// 计 算 柱状 图 区 间 
for ($i=0; $i < 12; $i++) { 
if(date('n')-$i == 0) { 
$month = 12; 
S$year = date('Y') - 1; 
} else { 
Smonth = date('n')-$i; 
S$year = date('Y'); 
| 


if($month > 9) { 
Sindex = $year.'/'.$month; 
$barData[$index] = array('income'=>0, 'expense'=>0); 


} else { 

Sindex = $year.'/0' .$month; 

$barData[$index] = array('income'=>0, 'expense'=>0); 
} 


1/ 计算 总 收入 、 分 月 收入 和 各 收入 分 类 小 计 
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Sincome = new Income(); 

Srs = $income->retrieve many('uid = ? AND 
UNIX TIMESTAMP (mod dt)>UNIX TIMESTAMP (?) "， 
array($ SESSION['uid'],date("Y-m-01"， 
strtotime ('1 year ago')))); 

StotalIncome = 0; 


$incomeDetails array('Salary & Wages' => 0, 
"Bonus' => 0, 
"Dividend' => 0, 
"Rentals' => 0, 
"Others' => 0); 
f(srs)t 


foreach (Srs as $index => S$object) { 
$amount = $object->rs['amount']; 
// 总 收入 


StotalIncome += $amount; 


// 分 月 收入 
$barData[date('Y/m' ,strtotime ($0object->rs['mod dt']))] 
['income'] += $amount; 


// 各 收入 分 类 小 计 
switch ($object->rs['category']) { 
case "SW": 
$incomeDetails['Salary & Wages'] += $amount; 
break; 
case 'BN': 
$incomeDetails['Bonus'] += $amount; 
break; 
case 'DD': 
$incomeDetails['Dividend'] += $amount; 
break; 
Case “RE: 
$incomeDetails['Rentals'] += $amount; 
break; 
Cans “OP's 
$incomeDetails['Others'] += $amount; 
break; 


} 
// 计 算 总 支出 、 分 月 支出 和 各 收入 分 类 小 计 


Sexpense = new Expense(); 

$rs = $expense->retrieve many('uid = ? AND 
UNIX TIMESTAMP (mod dt) > UNIX TIMESTAMP(?2)', 
array($ SESSION["'uid'],date('Y-m-01', strtotime 
('1 year ago')))); 

StotalExpense = 0; 


$expenseDetails array('Garment & Dressing' => 0, 
'Food & Vegetable' => 0, 
"Housing & Rentals' => 0, 
"Utilities & Others' => 0, 
'Transportation' => 0); 
if($rs){ 


foreach ($rs as $index => S$object) { 
$amount = $object->rs['amount"']; 


// 总 支出 


i 
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StotalExpense += $amount; 


// 分 月 支出 
$barData[ldate('Y/m', strtotime (S$object->rs["mod qt']))] 
['expense'] += $amount; 


// 各 收入 分 类 小 计 
Switch ($object->rs['category']) { 
case "GD': 
S$expenseDetails['Garment & Dressing'] += $amount; 
break; 
case "FD"': 
$expenseDetails['Food & Vegetable'] += $amount; 
break; 
case "HR" : 
S$expenseDetails["Housing & Rentals'] += $amount; 
break; 
case "UT'" : 
SexpenseDetails['Utilities & Others'] += $amount; 
break; 
case "TPE 
S$expenseDetails['Transportation'] += $amount; 
break; 


} 


S$Sview->set ('totalIncome', S$totalIncome); 
Sview->set ('totalExpense', S$totalExpense); 
Sview->set ('balance', S$totalIncome-$totalExpense); 
Sview->set ('barData', $barData); 

Sview->set ('incomeDetails', $incomeDetails); 
Sview->set ('expenseDetails', $expenseDetails); 
Sview->set ('fullname', $fullname); 

Sview->set ('avatar', S$avatar); 

Sview->dump (); 


. 


function getWhiteSpace ($int){ 
$string ="'"; 
for ($i=0; $i < $int; $i++) { 
$string .= 'énbsp;"'; 
| 


return $string; 


人 


这 个 页 面 的 主要 业务 逻辑 是 ， 根 据 SESSION 变量 中 缓存 的 用 户 ID， 获 取 用 户 一 年 内 
的 所 有 收入 和 支出 记录 ， 并 根据 这 些 记录 计算 其 总 收入 、 总 支出 、 节 余 及 绘制 分 月 收入 / 
支出 柱状 图 和 分 类 小 计 圈 图 。 大 家 可 以 重点 关注 一 下 , 我 们 是 如 何 定义 柱状 图 的 时 间 范 围 、 
如 何 计算 分 月 收入 以 及 如 何 计算 分 类 小 计 的 部 分 。 读 懂 这 几 个 部 分 可 以 帮助 我 们 加 深 对 数 
据 的 理解 。 运 行 后 的 效果 如 图 13-14 所 示 。 
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1603.49 1809.63 


Toualspent Balance 


图 13-14 ”控制 台 页 面 的 布局 


13.4 习 题 


(1) 请 补充 13.3.1 小 节 中 规划 的 “修改 指定 的 收入 /消费 记录 ”。 
(2) 请 补充 13.3.1 小 节 中 规划 的 “删除 指定 的 收入 /消费 记录 ”。 
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第 14 章 常见 开源 的 PHP 应 用 


正 是 由 于 PHP 的 开源 特性 和 活跃 的 社 群 支持 ， 基 于 PHP 的 开源 应 用 层出不穷 。 在 本 
章 里 ， 我 们 就 来 了 解 一 下 当前 比较 稳定 、 定 期 更 新 的 PHP 应 用 ， 这 些 应 用 涉及 到 网 络 的 各 
方 各 面 。 如 果 有 兴趣 的 话 , 大 家 可 以 在 网 上 搜索 一 篇 名 为 “Top Ten Open Source PHP Apps” 
的 文章 。 这 篇 文章 :里 列 出 了 PHP 在 博客 、 论 坛 、 内 容 管理 、 百 科 、 社 交 类 应 用 (Digg 和 
微 博客 ) 、 图 片 管理 、RSS 和 电子 商务 等 方面 的 应 用 。 在 本 章 里 ， 对 博客 系统 、 内 容 管理 
系统 (CMS) 和 论坛 系统 的 开源 PHP 应 用 做 相关 的 介绍 。 


14.1 WordPress 


WordPress 是 当今 最 受 欢 迎 的 基于 PHP 的 开源 博客 系统 。 按 照 WordPress 官方 网 站 的 
介绍 ，WordPress 也 是 一 种 内 容 管理 系统 。 而 维基 百科 对 内 容 管理 系统 的 定义 是 “一 种 用 
于 组 织 和 促进 共同 内 容 创造 的 系统 ”。 对 于 WordPress 这 个 名 字 的 由 来 ， 最 显而易见 的 原 
因 就 是 Word+Press, Word 代表 着 构成 内 容 的 基本 元 素 , 而 Press 则 代表 着 内 容 创造 的 方式 。 
通过 Word 和 Press 的 结合 ， 实 现 协同 的 内 容 创造 。 

WordPress 应 用 程序 有 两 个 部 分 构成 ， 一 个 为 前 台 ， 另 一 个 为 后 台 。 所 谓 前 台 就 是 广 
大 互联 网 用 户 访 问 利 用 WordPress 应 用 程序 搭建 的 个 人 博客 系统 时 可 以 看 到 的 内 容 ， 而 所 
谓 后 台 则 是 只 有 内 容 贡献 者 才 可 以 查看 到 的 部 分 。 前 台 可 以 实现 的 主要 功能 包括 浏览 和 基 
于 留言 的 用 户 交互 。 前 台 的 表现 形式 多 样 ， 我 们 可 以 为 前 台 部 署 极 富 个 性 化 的 主题 ， 从 而 
使 得 每 一 个 利用 WordPress 应 用 程序 搭建 的 个 人 博客 不 会 千 人 一 面 。 后 台 可 以 实现 的 主要 
功能 则 包括 分 类 管理 、 博 客 管理 和 主题 管理 等 ， 有 着 强大 的 管理 功能 ， 可 以 帮助 博客 运营 
者 充分 展示 其 想 要 与 大 家 分 享 的 内 容 。 图 14-1 展示 了 WordPress 的 后 台 。 


a Dashboard 、 


图 14-1 WordPress 的 后 台 


1 这 篇 文章 的 中 文 版 可 以 在 http:/www.williamlong info/archives/1852.html 看 到 。 
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14.1.1 安装 WordPress 


在 WordPress 的 官方 网 站 上 ， 我 们 可 以 看 到 关于 WordPress 安装 的 介绍 。 如 果 你 和 我 
一 样 是 在 本 地 测试 WordPress 应 用 程序 ， 那 么 可 以 将 下 载 到 的 WordPress 包 文件 (.zip) 解 
压 到 本 地 Web 服务 器 根 文件 夹 下 。 

随后 打开 浏览 器 ,在 地 址 栏 中 输入 http://localhost。 如 果 在 浏览 器 中 看 到 了 类 似 于 如 图 
14-2 所 示 的 内 容 ， 表 明 WordPress 可 以 在 本 地 服务 器 上 安装 了 。 


图 14-2 WordPress 安装 程序 配置 文件 缺失 提示 


这 时 ， 我 们 需要 先 为 WordPress 应 用 程序 配置 一 个 MySQL 数据 库 。 打 开 在 第 2 章 安 
装 的 PHPMyAdmin。 使 用 root 账户 登录 后 ， 在 首页 的 “新 建 数据 库 ” 下 的 文本 框 内 输入 
“wordpress”， 并 在 其 后 的 下 拉 列 表 框 内 选择 “utf8-general-ci”， 然 后 单 击 “创建 ”按钮 
创建 数据 库 。 

当 界 面 上 提示 “创建 数据 库 Wordpress 成 功 ” 时 ， 就 可 以 关闭 PHPMyAdmin 了 。 

这 时 ， 我 们 回 到 图 14-2 所 示 的 页 面 ， 单 击 左 下 角 的 “Creat a Configuration File” 按 钮 ， 
开始 创建 WordPress 配置 文件 ， 如 图 14-3 所 示 。 


(YWworpprrss 


图 14-3” WordPress 安装 程序 首页 
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按照 首页 上 的 提示 ， 我 们 需要 向 服务 器 提交 一 些 信息 。 所 需 信息 包括 “数据 库 名 ”、 
“数据 库 用 户 名 ”、“ 数 据 库 密码 ”、“ 数 据 库 所 在 主机 ”以 及 “数据 库 表 前 缀 ”。 由 于 我 
们 是 在 本 地 进行 测试 ， 数 据 库 信 息 如 下 。 
口 数据 库 名 : wordpress; 
口 数据 库 用 户 名 : root; 
口 数据 库 密码 : PasswOrd*; 
口 数据 库 所 在 主机 : localhost; 
口 数 所 库 表 前 级 : wp _。 
当 我 们 看 到 如 图 14-4 所 示 的 信息 时 ， 表 明 数 据 库 连接 成 功 ， 可 以 正式 开始 安装 了 。 


(WWorDPrEss 


Al right sparky! You've made ft through this part of the instalation Worcpress can now communicate with your 
database If you are ready. time now to 


Run me nstall 


图 14-4 ”数据 库 连接 成 功 


单 击 “Run the install” 按 钮 ， 开 始 安 装 。 当 出 现 如 图 14-5 所 示 的 表单 时 ， 表 明 数 据 库 
安装 完毕 ， 可 以 开始 配置 博客 信息 了 。 


Readgue 
ay to usng the 
Please proyide the fowng nformaton Don laorny you can aways change these Setngs Haler 
Site THe 1 
Usermame am 
wreves carp rre em eaererere ous woeces venaowm nm 0 


图 14-5 配置 博客 信息 


在 这 里 ， 我 们 需要 设置 的 参数 有 : “网 点 名 称 ”、“ 用 户 名 ”、“ 密 码 (输入 两 次 ) ” 
和 “电邮 地 址 ”等 。 由 于 我 们 是 在 本 地 进行 测试 ， 外 网 无 法 访问 。 所 以 这 些 信息 可 以 随 性 
输入 。 

输入 的 参数 分 别 如 下 。 

口 网 站 名 称 : WordPress; 
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口 用 户 名 : admin (默认 值 》 了 
口 密码 : PasswOrd*; 

口 电邮 : postmaster@localhost.com。 

然后 单 击 下 方 的 “Complete” 按 钮 完成 配置 。 随 后 我 们 会 看 到 提示 成 功 的 页 面 。 单 击 
页 面 左 下 角 的 “Log In” 按 钮 ， 然 后 输入 之 前 配置 的 用 户 名 和 密码 ， 即 可 登录 WordPress 
后 台 。 在 后 台 的 左上 角 WordPress 图 标 旁边 显示 了 博客 的 名 称 “WordPress”。 将 鼠标 指针 
移 上 去 ， 然 后 在 下 拉 菜 单 中 单 击 “Visit Site” 按 钮 ， 即 可 看 以 站 点 的 前 台 。 

至 此 ，WordPress 的 安装 成 功 。 


14.1.2 ”使 用 QuickPress 发 布 一 条 博客 


在 首页 右 下 方 单 击 “Site Admin ”按钮 回 到 后 台 ， 在 首页 的 Dashboard 上 找到 
“QuickPress” 面 板 。 在 该 面板 中 有 三 个 文本 框 ， 分 别 为 “博客 标题 ”、“ 博 客 内 容 ” 和 “ 博 
客 标签 ”。 在 “博客 标题 ”和 “博客 内 容 ” 两 个 文本 框 之 间 有 一 个 “Add Media” 的 按钮 ， 
单 击 它 可 以 上 传 多 媒体 文件 到 库 中 ， 或 直接 从 库 中 选择 需要 插入 博客 中 的 文件 。 在 “博客 
标签 ”文本 框 下 方 有 三 个 按钮 “Save Draft”、“Reset” 和 “Publish”， 分 别 代 表 保 存 草 
稿 、 重 置 和 直接 发 布 。 

在 第 一 个 文本 框 中 输入 “My First Post”， 然 后 单 击 “Add Media” 按 钮 上 传 了 一 张 图 
片 ， 接 着 在 下 面 的 文件 框 里 写 上 一 句 话 ， 然 后 再 为 这 条 博客 打上 一 个 标签 ， 单 击 “Publish” 
按钮 直接 发 布 博客 ,如 图 14-6 所 示 。 当 QuickPress 面板 里 出 现 一 条 黄 底 的 提示 信息 说 明博 
客 发 布 成 功 时 ， 我 们 可 以 单 击 其 后 的 “View Post” 链 接 查 看 该 博客 ， 或 者 单 击 “Edit Post” 
链接 修改 该 博客 。 


QuidPress 


Pest pualished View post| Edit post 


: You can also tr Press This. easy blool 3 
QuickPress oucan also tr Press This. easy bloooina fiom anwhere on he Web. 
My First Post 

到 AldNeda I 


= 00 hon 300 dass=alignnone szemedium wp-imace- | 
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图 14-6 ”使 用 QuickPress 发 布 一 条 博客 


现在 我 们 单 击 “View Post” 按 钮 查看 已 发 布 的 博客 ， 就 能 在 前 台 看 到 这 条 博客 的 样子 
了 。 发 布 的 这 条 博客 如 图 14-7 所 示 。 


14.1.3 ”修改 已 发 布 的 博客 


在 查看 完 刚 刚 发 布 的 博客 之 后 ， 发 现 有 些 地 方 不 太 满 意 ? 那 我 们 就 回去 再 改 改 。 单 击 
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页 面 右 侧 的 “Site Admin” 链 接 回 到 后 台 。 在 后 台 的 左 侧 有 一 列 菜单 ， 其 中 第 二 个 名 为 
“Posts”， 单 击 这 个 菜单 按钮 ， 进 入 “Posts” 管 理 页 面 ， 如 图 14-8 所 示 。 


My First Post 


图 14-7 我 的 第 一 条 博客 


图 14-8 博客 管理 页 面 


在 这 里 ， 我 们 可 以 看 到 一 张 包含 了 所 有 已 发 布 、 未 发 布 的 博客 的 列表 。 现 在 这 张 表 里 
有 两 条 博客 ， 第 一 条 就 是 我 们 刚刚 发 布 的 。 鼠 标 移 上 去 的 时 候 ， 在 这 条 博客 的 标题 下 面 出 
现 了 四 个 链接 “Edit”、“Quick Edit”、“Trash” 和 “View”， 分 别 用 于 编辑 、 快 速 编 
辑 、 删 除 和 查看 该 博客 。 

在 这 里 我 们 单 击 “Edit” 编 辑 该 博客 ， 很 快 便 进入 了 如 图 14-9 所 示 的 博客 编辑 页 面 。 

在 该 页 面 上 ， 我 们 可 以 修改 此 博客 的 任何 内 容 ， 包 括 它 的 名 称 、 内 容 、 所 属 类 别 、 发 
布 时 间 、 适 用 范围 和 格式 等 等 。 在 这 里 ， 我 们 只 修改 一 下 博客 的 格式 : 在 右 侧 的 “Format” 
面板 中 选择 “Image”， 然 后 单 击 上 方 的 “Update” 按 钮 ， 更 新 博客 。 

再 次 查看 该 博客 时 ， 发 现 博客 上 方 的 大 字 标 题 没有 了 ， 变 成 了 图 片 下 方 的 几 行 小 号 的 
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文字 ， 图 片 在 页 面 中 变 得 更 加 突出 了 。 


p+ pr 
办 Edit Post suren ET 
My First Post a 
a see Freiew Cranges 
me Er 
i B11~ 主 上 壮志 成 前 情 泡 - 四 目 
图 14-9 编辑 博客 


14.1.4 ”定制 页 面 


在 WordPress 中 ， 与 博客 类 似 的 元 素 叫做 页 面 (Page) 。 在 前 台 看 来 ， 页 面 和 博客 没 
有 什么 不 同 。 但 是 在 后 台 ， 页 面 是 可 以 被 直接 插入 到 导航 菜单 中 的 ， 而 博客 则 只 能 通过 其 
URL 来 实现 。 为 了 对 定制 的 页 面 有 个 明确 的 概念 ，WordPress 安装 之 后 会 预 置 一 个 名 为 
“Sample Page” 的 定制 的 页 面 ， 并 将 其 放 在 了 导航 菜单 中 。 

单 击 前 台 首 页 导航 菜单 中 的 “Sample Page”， 会 发 现 它 跟 我 们 之 前 发 布 的 博客 的 确 没 
有 什么 区 别 。 回 到 后 台 , 在 页 面 左 侧 的 导航 树 中 单 击 “Pages” 菜单 按钮 之 后 进入 页 面 管 理 ， 
如 图 14-10 所 示 。 在 这 里 ， 我 们 可 以 查看 系统 中 现 有 的 页 面 ， 并 可 以 像 修 改 博客 一 样 修改 
它们 。 

这 里 ， 我 们 可 以 把 “Sample Page ”修改 一 下 ， 将 其 名 称 修改 为 “Contact Us”。 


0 re Anthor pt 
ES Phugine i i 
全 un 
Th Tooks 
回 Semings 


图 14-10 管理 页 面 
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将 鼠标 移动 到 “Sample Pages” 上 ， 随 后 在 出 现 的 四 个 链接 中 ， 单 击 “Quick Edit” 按 
钮 ， 出 现 如 图 14-11 所 示 的 “快速 编辑 ”表单 。 这 张 表 单 大 体 上 与 博客 的 “快速 编辑 ” 相 
同 ， 如 图 14-11 所 示 。 


口 Te uthor bd Date 
OUICK EDT 


-op- OPrhate 


图 14-11 快速 编辑 页 面 


我 们 将 表单 中 的 “Title” 文 本 框 的 内 容 由 “Sample Pages” 修 改 为 “Contact Us”， 然 
后 单 击 右 下 方 的 “Update ”按钮 完成 修改 。 这 时 ,我 们 再 返回 前 台 , 发 现 导 航 菜 单 中 的 “Sample 
Page” 已 经 被 替换 成 了 “Contact Us”。 


14.1.5 添加 博客 分 类 


回 到 前 台 后 ， 我 们 发 现 之 前 发 布 的 名 为 “My First Post” 的 博客 所 属 的 分 类 为 
“Uncategorized”。 这 表明 该 博客 并 未 归 入 适当 的 分 类 中 。 在 本 小 节 里 , 我 们 将 为 WordPress 
站 点 添加 几 个 分 类 ， 以 便 更 好 地 管理 博客 内 容 。 

单 击 页 面 右 侧 的 “Site Admin” 返 回 后 台 ， 在 左 侧 的 菜单 列表 中 ， 单 击 “Posts” 菜 单 
按钮 。 然 后 单 击 “Categories” 链 接 进 入 博客 分 类 管理 页 面 ， 如 图 14-12 所 示 。 


图 14-12 博客 类 别管 理 


在 这 里 我 们 可 以 添加 新 的 分 类 ， 还 可 以 查看 、 修 改 和 删除 已 有 分 类 。 需 要 注意 的 是 ， 
预 置 分 类 “Uncategorized” 可 以 修改 ， 但 不 可 以 删除 。 如 需 移 除 一 个 自 定义 的 分 类 ， 需 要 
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先 将 其 中 的 博客 移 至 其 他 的 分 类 中 。 

在 这 里 ， 我 们 打算 添加 “Writing Project”、“Graphic Design” 和 “Life Drops” 三 个 
分 类 ， 然 后 将 “Uncategorized” 修 改 为 “Warm Greetings”。 在 添加 和 修改 博客 分 类 时 ， 需 
要 注意 一 个 名 为 “slug” 的 参数 的 设置 ， 该 参数 是 为 了 帮助 浏览 器 更 好 地 识别 分 类 的 名 称 。 
因此 不 要 在 slug 参数 中 使 用 空格 和 其 他 特殊 字符 。 我 的 建议 是 ， 使 用 与 分 类 名 称 相同 的 内 
容 。 如 果 分 类 名 称 中 有 空格 ， 用 “-” 蔡 代 空 格 。 如 果 分 类 名 称 中 有 大 写字 母 ， 在 slug 参 
数 中 使 用 相应 的 小 写字 母 ， 如 图 14-13 所 示 。 


Ci 


Add New Category Buk Adions 国 | sw 

N 口 Nama Deseription ug Posts 
口 SGmphic Desig phicaesig 
OWriting Project wrting-project 0 


Eotl ourEaN Delete 


图 14-13 添加 新 分 类 
至 止 ， 我 们 完成 了 博客 分 类 的 添加 和 修改 。 


14.1.6 ”管理 导航 菜单 


在 添加 了 博客 、 页 面 和 分 类 之 后 , 为 了 让 访问 博客 的 人 可 以 更 方便 地 找到 想 要 的 内 容 ， 
我 们 打算 把 各 分 类 和 页 面 做 为 菜单 项 添加 到 导航 菜单 中 。 

单 击 页 面 左 侧 的 “Appearance ”菜单 按钮 ， 然 后 单 击 其 下 的 “Menus” 链 接 进 入 菜单 
管理 页 面 ， 如 图 14-14 所 示 。 


图 14-14 菜单 管理 
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页 面 的 左 侧 为 可 向 菜单 中 添加 的 元 素 ， 它 们 一 共有 三 种 : 自 定义 链接 、 定 制 页 面 和 博 

客 分 类 。 这 意味 着 我 们 可 以 在 导航 菜单 中 添加 自 定义 链接 、 定 制 好 的 页 面 和 博客 分 类 。 页 
i 的 右 侧 为 菜单 编辑 区 域 。 在 编辑 菜单 之 前 ， 我 们 需要 创建 一 个 新 的 菜单 ， 可 以 创建 的 菜 
单数 量 与 当前 使 用 的 主题 有 关 。WordPress 当前 版 本 的 默认 主题 只 支持 一 个 菜单 。 
在 填写 了 菜单 名 称 之 后 ， 单 击 “Create Menu” 按 钮 创建 菜单 ， 然 后 将 之 前 创建 好 的 页 
面 和 博客 分 类 添加 到 菜单 中 ， 并 通过 拖 钨 为 它们 排序 。 当 菜单 元 素 排列 妥当 之 后 ， 单 击 右 
下 角 的 “Save Menu” 按 钮 保存 菜单 。 随 后 页 面 上 方 出 现 黄 底 通知 ， 告 诉 我 们 保存 菜单 成 
功 了 。 

现在 就 让 我 们 返回 WordPress 前 台 查 看 一 下 导航 菜单 的 效果 吧 。 首 页 上 赫然 出 现 了 刚 
才 添加 的 元 素 ， 单 击 各 个 菜单 按钮 ， 可 以 进入 相应 的 博客 分 类 和 自 定 义 页 面 ， 如 图 14-15 
所 示 。 很 炫 吧 ! 


WordPress 


图 14-15 添加 了 导航 菜单 的 首页 


14.1.7 管理 前 台 主 题 


WordPress 默认 的 主题 ， 比 较 适 合 二 次 开发 。 如 果 不 想 亲 手 制 作 WordPress 主题 的 话 ， 
也 会 有 很 多 的 选择 。 单 击 页 面 右 侧 的 “Site Admin ”按钮 回 到 系统 后 人 台 ， 青 单 击 左 侧 的 
“Appearance” 菜 单 按钮 ， 进 入 主题 管理 页 面 ， 如 图 14-16 所 示 。 

在 主题 管理 页 面 上 ， 有 两 个 页 签 。 一 个 为 “Manage Themes”， 用 于 管理 主题 ， 另 一 
个 为 “Install Themes”， 用 于 安装 新 主题 。 我 们 进入 第 二 个 页 签 。 然 后 单 击 页 签 下 方 的 一 
排 链接 中 的 “Newest” 查 看 最 新 发 布 的 主题 。 在 每 个 主题 的 截图 下 方 会 出 现 三 个 按钮 ， 分 
别 是 “Install Now”、“Preview” 和 “Details”。 读 者 可 以 使 用 这 些 链 接 来 安装 、 预 览 或 
者 详细 查看 对 应 的 主题 。 当 选中 一 个 主题 后 ， 安 装 它 ， 然 后 再 返回 “Manage Themes” 页 
签 激活 它 。 再 返回 前 台 看 看 效果 吧 ， 图 14-17 所 示 就 是 我 应 用 了 新 主题 之 后 的 效果 ， 是 不 
是 大 不 一 样 了 呢 ? 
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BOLDR 


图 14-16 主题 管理 


较 WARM GREETINGS 。 WRITING PROJECT GRAPHIC DESIGN CONTACTUS 


MY FIRST POST 


POSTD IN WARM GREFTINGS. 
RECENT POSTS 


MY FIRST POST 
HELOWORLDI 


RECENT COMMENTS 


MRWORDPRESS ON HELIO 
WORLD! 


14.1.8 小 结 


图 14-17 应 用 主题 后 的 效果 


WordPress 做 为 当今 世界 上 最 流行 的 博客 软件 ， 其 稳定 性 和 丰富 的 扩展 性 是 毋庸 置疑 


的 。 如 果 读 者 想 要 搭建 一 
二 之 选 。 
同时 WordPress 也 支 


个 个 人 博客 ， 分 享 你 的 所 见 、 所 闻 及 所 想 ，WordPress 绝对 是 不 


竺 二 次 开发 ， 也 可 以 结合 本 书 中 学 到 的 PHP 知识 和 WordPress 网 


站 上 的 资料 ， 发 挥 多 看 、 多 问 和 多 想 的 “三 多 ”精神 ， 为 WordPress 开发 新 的 主题 和 插件 。 


如 果 有 可 能 的 话 ， 我 也 打 
共享 。 


算 总 结 一 下 自己 在 WordPress 主题 和 插件 开发 方面 的 经 验 与 大 家 
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14.2 Drupal 


提起 Drpal， 大 名 如 雷 贯 耳 。 它 是 使 用 PHP 语言 编写 的 开源 内 容 管 理 框架 CCMF) ， 
它 由 内 容 管理 系统 CMS) 和 PHP 开发 框架 (Framework) 共同 构成 。 连 续 多 年 荣获 全 球 
最 佳 CMS 大 奖 ， 是 基于 PHP 语言 最 著名 的 Web 应 用 程序 。 

Drupal 是 一 套 开 源 系 统 ， 全 球 数 以 万 计 的 Web 开发 专家 都 在 为 Drupal 技术 社区 贡献 
代码 。 因 此 ，Drupal 的 代码 在 安全 性 和 健壮 性 上 具有 世界 最 高 水 平 。 这 也 是 美国 白宫 、 美 
商务 部 、 法 国政 府 、 纽 约 时 报 、SONY 等 著名 政府 和 机 构 纷 纷 采 用 Drupal 建设 网 站 的 最 
重要 的 原因 。 

但 是 , 与 WordPress 等 使 用 模板 建站 不 同 , Drupal 的 学 习 曲 线 相当 漫长 和 陡峭 , 比 PHP 
难度 大 得 多 ,要求 也 高 得 多 。 事实 上 , 只 有 精通 XHTML、CSS、Javascript、 PHP 和 MySQL 
的 开发 人 员 ， 经 过 长 期 刻苦 的 学 习 ， 才 有 可 能 真正 的 驾驭 Drupal。 

在 本 节 里 ， 我 们 将 简单 地 介绍 一 下 Drupal 的 安装 和 使 用 。 


ll 


14.2.1 安装 Drupal 


在 Drupal 的 官方 网 站 上 ， 我 们 可 以 看 到 关于 Drupal 安装 的 介绍 。 如 果 你 和 我 一 样 是 
在 本 地 测试 Drupal 应 用 程序 ， 那 么 可 以 将 下 载 到 的 Drupal 包 文件 (.zip) 解压 到 本 地 Web 
服务 器 根 文件 夹 下 。 

随后 打开 浏览 器 ,在 地 址 栏 中 输入 http://localhost。 如 果 在 浏览 器 中 看 到 了 类 似 于 如 图 
14-18 所 示 的 内 容 ， 表 明 WordPress 可 以 在 本 地 服务 器 上 安装 了 。 


Select an installation profile 


e 


， choose pronle 


图 14-18 ”Drupal 安装 起 始 页 面 


它 的 安装 过 程 与 WordPress 类 似 ， 都 需要 在 本 地 先 为 其 创建 一 个 MySQL 数据 库 。 因 
此 ,我 们 再 次 打开 PHPMyAdmin, 创建 一 个 名 为 Drupal 的 数据 库 , 然后 关闭 PHPMyAdmin。 
现在 我 们 回 到 如 图 14-18 所 示 的 页 面 ， 选 择 “Standard” 标 准 安装 模式 ， 单 击 “Save and 
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Continue” 按 钮 保存 当前 设置 并 继续 。 

按照 页 面 提示 ， 下 载 中 文 语言 包 ， 并 将 其 放 在 指定 位 置 ， 然 后 重新 加 载 当前 页 面 使 设 
置 生 效 。 当 出 现 如 图 14-19 所 示 的 页 面 时 ， 选 择 “Chinese,Simplified (简体 中 文 )”， 然 
后 单 击 “Save and Continue” 按 钮 保存 设置 并 继续 。 


Choose language 


图 14-19 选择 Drupal 界面 语言 


这 时 ，Drupal 的 安装 界面 就 自动 变 成 简体 中 文 了 。 现 在 我 们 填 入 “数据 库 名 称 ”、“ 数 
据 库 用 户 名 ”和 “数据 库 密码 ”， 然 后 单 击 “ 保 存 并 继续 ”按钮 保存 当前 设置 并 继续 ， 如 
图 14-20 所 示 。 需 要 注意 的 是 ， 在 本 地 测试 Drupal 程序 使 用 默认 的 高 级 设置 即 可 。 若 在 远 
程 服务 器 上 测试 ， 且 MySQL 数据 库 与 PHP 不 在 同一 台 服 务 器 上 ， 则 有 可 能 需要 对 “高 级 
选项 ”里 的 “数据 库 主 机 ”和 “端口 ”进行 设置 。 


枚 据 库 设 澡 
4 凌志 库 关于 
人 WiQL NaiaDs， 起 首 尖刀 的 
再 于 pvow 区 所 伐 让 负 委 近 关 类 再， 让 的 Pvp 说 秆 亿 支 拆 一 个 间 拍 的 笋 迫 夺 类 型 ， 拓 LE 
全 自 吕 六 、 
L522 和 
v 一 
v Es 二 于 Ovow 闪 拓 和 亲 友 站 数据 吉 侦 而 和 加 上 公休 和 志 贡 半 光 库 ， 才 衣 站 人 安 要 
只 半 人才 
训话 下 抽 库 用 户 名。 
现下 本 到 
党 其 流 
保 六 并 外 从 


图 14-20 填写 数据 库 信息 
这 时 ，Drupal 就 开始 安装 了 。 安 装 程序 会 先 按 设 置 完成 安装 ， 然 后 安装 本 地 化 文件 。 
接着 ， 我 们 需要 手动 设置 一 下 网 站 ， 这 一 点 与 WordPress 相同 。 在 这 里 需要 设置 网 站 名 称 、 
网 站 邮箱 、 管 理 员 账户 、 服 务 器 所 在 地 和 当地 时 区 。 当 完成 这 些 设 置 后 ， 单 击 “ 保 存 并 继 
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续 ” 按 钮 ， 系 统 将 保存 当前 设置 并 完成 本 地 化 文件 的 安装 。 
当 我 们 看 到 如 图 14-21 所 示 的 页 面 时 ， 表 明 我 们 的 安装 成 功 了 。 这 时 ， 我 们 可 以 单 击 
页 面 上 的 “访问 新 网 站 ”链接 来 访问 安装 好 的 Drupal 应 用 程序 。 


Drupal 安装 完成 


图 14-21 Drupal 安装 成 功 


14.2.2 了 解 Drupal 的 使 用 方法 


在 打开 网 站 首页 后 , 系统 会 自动 以 你 在 安装 过 程 中 创建 的 管理 员 账 户 登录 。 如 果 没 有 ， 
你 可 以 手动 输入 用 户 名 和 密码 登录 Drupal。 当 成 功 登 录 Drupal 后 ， 你 会 发 现 首页 顶端 出 
现 了 一 条 黑 底 的 工具 条 ， 如 图 14-22 所 示 。 这 也 是 Drupal 与 WordPress 不 同 的 地 方 。 
WordPress 为 我 们 提供 了 一 个 独立 的 后 台 ， 如 果 需 要 在 完成 一 项 设置 之 后 查看 效果 ， 则 需 
要 手动 切换 到 前 台 查 看 。 而 在 Drupal 里 ,没有 所 谓 前 台 和 后 台 的 概念 。 如 果 你 是 以 管理 员 
账户 登录 的 , 则 可 以 在 页 面 的 顶端 看 到 相应 的 管理 导航 菜单 。 单 击 管理 导航 菜单 中 的 项 目 ， 
则 会 在 当前 页 面 上 方 出 现 了 一 悬浮 的 区 域 供 你 进行 管理 操作 。 所 有 的 操作 都 会 实时 更 新 到 
Drupal 站 点 中 ， 可 以 实时 看 到 设置 的 效果 。 


编 和 快捷 刍 


图 14-22 Drupal 的 管理 导航 菜单 


在 这 里 ， 我 们 可 以 选择 “报告 ”|“ 状 态 报告 ”命令 来 查看 当前 站 点 的 状态 。 图 14-23 
展示 了 安装 在 本 地 计算 机 上 的 Drupal 站 点 的 状态 报告 。 

在 这 份 报告 中 ， 可 以 看 到 Drupal 核心 的 版 本 、 服 务 器 上 安装 的 PHP 引擎 的 版 本 、 数 
据 库 系统 的 类 型 和 版 本 、 文 件 系 统 是 否 可 写 等 与 管理 Drupal 站 点 相关 的 内 容 ， 并 可 以 据 此 
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来 了 解 Drupal 站 点 的 运行 情况 ， 如 图 14-23 所 示 。 
单 击 悬浮 区 域 右上 角 的 “ 司 ” 按钮 就 可 以 关闭 悬浮 区 域 , 回 到 打开 悬浮 区 域 之 前 的 页 面 。 


的 EB 
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Unesce 医 
we 
上 全 浊 朗 

从 的 服务 痢 关 汉 避 的 区 以 县 示 廊 件 上 依 江 度 ， 才 莽 空 迹 PCLL 上 从 于 古 否 《 辣 造 ) 可 去 本 sr 
存 取 seduce oho 
固原 更 六 
背 穆 大 
数据 诛 和 入 版 太 


可 甬 需 理 夺 人 职 稀 厚生 三代 全 中 全 的 所 村 本 节 并 至 二 当前 楼 岂 及 议 四 基 近 权 限 。 如 
和， 生计 冯 可 能 会 论 上 一 漠 时 站 ， 在 重 寻 过 有 完 攻关， 及 宁 会 向 盐 使 有 新 约 权限 。 王道 内 


RP 


图 14-23 ”Drupal 立 


点 状态 报告 


14.2.3 ”管理 站 点 内 容 


单 击 管理 导航 菜单 中 的 “内 容 ” 节 点 ， 可 以 打开 “内 容 ” 乃 浮 窗 ， 如 图 14-24 所 示 。 


Sr wnbmastor 浊 


ET 


图 14-24 内 容 管理 
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在 “内 容 ” 悬 浮 窗 中 ， 我 们 可 以 对 站 点 现 有 的 内 容 和 与 之 相关 的 评论 进行 管理 ， 也 可 
以 添加 新 的 内 容 。 在 Drupal 中 ， 默 认 内 容 分 为 “基本 页 面 ” 和 “文章 ”两 种 类 型 。 读 者 也 
可 以 在 “结构 ”节点 下 的 “内 容 类 型 ”中 定义 新 的 类 型 。 单 击 “ 内 容 ” 悬 浮 窗 内 的 “添加 
内 容 ” 链 接 ， 然 后 单 击 “ 文 章 ” 创 建 一 篇 文章 。 这 时 ，“ 创 建文 章 ” 悬 浮 窗 替 代 了 “内 容 ” 
苦 浮 窗 ， 出 现在 我 们 面前 。 

“创建 文章 ”悬浮 窗 里 主要 有 两 个 部 分 。 其 一 为 文章 内 容 区 域 ， 如 图 14-25 所 示 ; 其 二 
为 文章 属性 区 域 ， 如 图 14-26 所 示 。 


图 14-25 文章 内 容 区 域 图 14-26 文章 属性 区 域 


在 文章 的 内 容 区 域 ， 我 们 可 以 指定 文章 的 标题 、 标 签 和 内 容 ， 这 一 点 与 在 WordPress 
中 添加 文章 是 类 似 的 。 如 果 需 要 为 文章 添加 摘要 ， 可 以 单 击 “Body” 旁 边 的 “编辑 摘要 ” 
链接 进行 添加 。 

在 文章 的 属性 区 域 ， 我 们 可 以 发 现 ， Re 与 WordPress 相 比 ， 有 了 比较 完善 的 版 本 
管理 系统 。 如 果 忠 实地 记录 每 一 次 对 文章 的 修订 ， 你 就 可 以 很 方便 地 恢复 之 前 的 记录 。 

现在 ， 我 们 创建 一 篇 文章 ， 标 题 为 《世界 ， wa 》， 然 后 到 Lorem Ipsum 网 站 上 复 
制 几 段 话 放 到 “Body” 文 本 框 中 。 单 击 悬 浮 窗 左 下 角 的 “保存 ”按钮 。 在 悬浮 窗 消失 后 ， 
我 们 就 能 看 到 《世界 ， 你 好 ! 》 这 篇 文章 被 放 到 了 首页 上 ， 同 时 在 文章 上 方 出 现 了 一 个 绿 
框 提 示 ， 告 诉 我 们 《世界 ， 你 好 ! 》 这 篇 文章 已 经 成 功 创建 了 ， 如 图 14-27 所 示 。 


Drupal 


a 世界 ， 你 好 ! 
EE 


图 14-27 成 功 创建 是 为“ 世界， 你 好 ! ”的 文章 


“0 


第 14 章 常见 开源 的 PHP 应 用 


1 于 ,我 们 现在 仍然 是 以 管理 员 的 身份 登录 Drupal 站 点 的 ， 因 此 ， 在 文章 标题 下 方 有 
两 个 页 签 ， 一 个 为 “查看 ”， 另 一 个 为 “编辑 ”。 如 果 对 文章 的 内 容 不 太 满 意 ， 可 以 单 击 
“编辑 ”页 签 ,在 弹出 的 悬浮 窗 中 对 文章 的 内 容 进 行 修改 。 现 在 ， 我 们 觉得 文章 的 倒数 第 二 
段 的 最 后 一 句 不 太 满意 ， 想 把 它 删除 。 那 么 ， 我 们 可 以 单 击 “ 编 辑 ” 页 签 ， 在 弹出 的 悬浮 


窗 中 删除 倒数 第 二 段 的 最 后 一 句 ， 然 后 在 文章 属性 区 域 的 “修订 版 本 信息 ”中 添加 对 本 次 
修订 的 描述 。 


当 单 击 悬 浮 窗 左 下 角 的 “保存 ”按钮 后 ， 悬 浮 窗 消 失 。 在 《世界 ， 你 好 ! 》 下 方 的 页 
签 数量 由 之 前 的 两 个 ， 变 成 了 现在 的 三 个 。 新 增加 的 页 签名 为 “修订 版 本 ”， 单 击 它 ， 我 
们 可 以 查看 之 前 的 修订 记录 。 在 “修订 版 本 ”页 签 中 ， 可 以 方便 地 恢复 或 者 删除 相应 的 记 
录 ， 如 图 14-28 所 示 。 


首页 > 世界 ,你 好 9 
修订 让 你 可 以 跟踪 内 容 多 个 版 本 之 间 的 差异 ， 以 及 恢复 到 早期 的 版 本 。 
修订 


webmaster 于 06/15/2013 - 09:40 


山 除 倒数 第 二 段 的 最 后 一 名 


webmaster 于 06/15/2013 - 09:38 


图 14-28 《世界 ， 你 好 ! 》 的 修改 版 本 记录 


添加 “基本 页 面 ” 与 添加 “文章 ”类 似 。 我 们 可 以 在 “基本 页 面 ”中 创建 一 个 名 为 
“关于 自己 ”的 页 面 ， 将 其 URL 别名 指定 为 “aboutme”。 具 体 的 操作 方法 这 里 就 不 再 
袭 述 。 


14.2.4 ”管理 站 点 结构 


大 家 或 许 已 经 发 现 了 ， 在 Drupal 的 “文章 ”编辑 区 域 并 没有 像 WordPress 一 样 为 我 们 
提供 一 个 可 以 方便 设置 分 类 的 选项 。 因 为 ，Drupal 对 站 点 的 管理 方式 更 加 灵活 。 而 通常 情 
况 下 ， 灵 活 就 意味 着 复杂 。 这 里 我 们 首先 需要 了 解 一 下 Drupal 的 站 点 结构 。 

Drupal 站 点 上 的 所 有 内 容 都 是 以 “节点 ”的 形式 存储 在 数据 库 中 的 。 也 就 是 说 ， 我 们 
发 布 的 每 一 篇 文章 ， 编 辑 的 每 一 个 页 面 、 乃 至 于 投票 、 论 坛 帖 子 等 等 都 是 一 个 个 的 节点 。 
这 一 点 大 家 可 以 从 上 一 小 节 里 发 布 的 文章 的 URL 中 看 到 。《 世 界 ， 你 好 ! 》 这 篇 文章 的 
URL 是 “http://localhost/node/1”， 也 就 是 说 《世界 ， 你 好 ! 》 这 篇 文章 是 当前 Drupal 站 
点 中 的 第 一 个 节点 。 以 后 我 们 创建 的 各 种 文章 也 好 、 投 票 也 好 、 论 坛 帖子 也 好 ， 怎 么 给 它 
们 分 类 呢 ? 

现在 ， 我 们 单 击 管理 菜单 上 的 “结构 ”， 打 开 “ 结 构 ” 悬 浮 窗 ， 如 图 14-29 所 示 。 

在 “结构 ” 翘 浮 窗 中 ， 我 们 可 以 对 Drupal 站 点 的 “内 容 类 型 ”、“ 分 类 ”、“ 区 块 ” 
和 “菜单 ”进行 管理 。 


“Ms 
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给 你 的 站 点 乱 加 新 菜单 、 编 辑 现 有 荣 单 、 以 及 重 命 名 和 重新 排列 


图 14-29 Drupal 站 点 的 “结构 ”悬浮 窗 


为 了 给 发 布 的 文章 分 类 ， 我 们 得 先 建立 一 个 分 类 。 在 “结构 ”悬浮 窗 上 单 击 “ 分 类 ” 
按钮 ， 然 后 在 出 现 的 “分 类 ”悬浮 窗 里 单 击 “ 添 加 词汇 表 ” 链 接 为 Drupal 站 点 添加 一 个 词 
汇 表 。 词 汇 表 名 称 确 定 为 “类 别 ”、 描 述 为 空 。 创 建成 功 后 ， 系 统 自动 返回 “分 类 ”悬浮 
窗 。 这 时 窗口 中 的 列表 里 出 现 了 刚才 添加 的 名 为 “类 别 ” 的 词汇 表 ， 同 时 ， 窗 口 的 上 部 也 
出 现 了 一 条 绿 框 黄 底 的 提示 信息 ， 告 诉 我 们 词汇 表 创建 成 功 了 ， 如 图 14-30 所 示 。 


由 
@ VTAEE A 


并 于 站 容 折 归 活 。 坏 河 由 动 末 六 和 根 纵 ，HK30 ， 一 个 对 全 水果" 9 天 可 于 但 全 趟 洒 "呆滞 哺 瑞 " 


中 沸 加 全 江天 


次 


图 14-30 添加 词汇 表 


现在 ， 单 击 “ 类 别 ” 行 右 侧 的 “添加 术语 ”在 类 别 下 新 建 一 些 类 别 。 下 面 是 我 打算 
添加 的 一 些 文章 的 分 类 以 及 它们 的 权重 和 URL 别名 ， 如 表 14-1 所 示 。 所 谓 “权重” 指 的 
其 实 是 它们 排列 的 先后 顺序 ， 而 URL 别名 ， 则 可 以 让 我 们 很 方便 地 通过 URL 找到 这 些 
分 类 。 


表 14-1 当前 Drupal 站 点 所 需 文章 分 类 


权 重 分 类 名 称 URL 别名 
0 平面 设计 graphdesign 
1 frontend 
2 lifedrop 


添加 完 这 些 分 类 后 ， 回 到 “类 别 ” 甚 浮 窗 ， 我 们 就 可 以 看 到 之 前 为 “类 别 ” 词 汇 表 添 
加 的 几 个 分 类 了 ， 如 图 14-31 所 示 。 


.394 
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到 开征 > 半 交 > 分 


本 rR 在. 


以 挝 注 村 四 折 六 区 拓 人 在 才 87P 的 相通 站 在 妈 本 贡 上 下 报 基 汪 好人 林 理 ， 


图 14-31 为 “类 别 ” 词 汇 表 添加 术语 


由 于 我 们 为 每 个 术语 都 设 定 了 一 个 URL 别名 , 因此 , 可 以 通过 URL 别名 来 访问 它们 。 
以 “平面 设计 ”为 例 ， 我 们 可 以 在 浏览 器 的 地 址 栏 中 输入 “http://localhost/graphdesign” 来 
访问 。 同 理 ， 也 可 以 分 别 使 用 “http://localhost/frontend” 和 “http://localhost/lifedrop ”来访 
问 “ 前 端 开发 ”和 “生活 点 滴 ” 两 个 分 类 。 

好 了 ， 现 在 我 们 需要 修改 一 下 “文章 ”类 型 的 表单 结构 ， 以 便 在 添加 新 文章 的 时 候 可 
以 为 其 分 类 。 单 击 管理 导航 菜单 上 的 “结构 ”， 然 后 单 击 “ 结 构 ” 蝶 浮 窗 里 的 “内 容 类 型 ” 
列 出 当前 系统 中 所 有 的 内 容 类 型 。 找 到 “文章 ”一 行 ， 单 击 其 右 侧 的 “管理 字段 ”链接 进 
入 “文章 ”悬浮 窗 的 “管理 字段 ”页 签 。 

在 该 页 签 里 的 “添加 新 字段 ”下 方 的 文本 框 里 输入 “articlegroup”， 然 后 将 “字段 类 
型 ”设置 为 “术语 来 源 ”、 控 制 设置 为 “选择 列表 ”。 接 着 单 击 下 方 的 “保存 ”按钮 保存 
当前 设置 ， 如 图 14-32 所 示 。 


i EE 
了 本 DO 下 
Ea 


入 
中 
和 
= 
全 
名 


图 14-32 ”为 “文章 ”类 型 的 内 容 添加 新 字段 


然后 在 弹出 的 “articlegroup” 葵 浮 窗 中 为 该 字段 指定 名 为 “类 别 ” 的 词汇 表 。 这 也 是 
我 们 之 前 创建 的 那个 词汇 表 , 单 击 “ 保 存 字 段 设置 ”按钮 。 这 时 , 系统 会 返回 “Article Group” 
悬浮 窗 ， 可 以 进一步 设置 这 个 字段 。 例 如 ， 我 们 可 以 设置 该 字段 为 必 填 项 ， 默 认 值 为 “ 生 
活 点 滴 ” 等 。 

至 此 ， 文 章 分 类 我 们 就 处 理 好 了 。 

现在 ， 我 们 想 把 之 前 创建 的 那 篇 文章 放 到 “生活 点 滴 ” 栏 目下 ， 应 该 如 何 操作 呢 ? 
单 击 管理 导航 菜单 上 的 “内 容 ”, 在 “内 容 ” 悬 浮 窗 下 方 的 列表 里 找到 《世界 ， 你 好 ! 》 


“es 
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这 篇 文章 。 单 击 其 右 侧 的 “编辑 链接 按钮 进入 文章 编辑 页 面 。 拖 动 页 面 找 到 “Article Group” 
字段 ， 选 择 下 拉 列 表 中 的 “生活 点 滴 ”， 然 后 单 击 “ 保 存 ” 按 钮 保存 当前 修改 。 

接着 我 们 在 浏览 器 的 地 址 栏 中 输入 “http://localhost/ifedrop” 来 访问 生活 点 滴 栏目 。 会 
发 现 《 世 界 ， 你 好 ! 》 这 篇 文章 已 经 安静 地 出 现在 “生活 点 滴 ” 这 个 标题 下 。 

可 是 有 一 个 问题 ， 这 样 访问 各 个 分 类 会 不 会 麻烦 了 一 些 。 是 否 可 以 把 这 些 分 类 添加 到 
主 菜单 中 呢 ? 答案 自然 是 肯定 的 。 
单 击 管理 导航 菜单 上 的 “结构 ”按钮 。 在 “结构 ” 蝶 浮 窗 中 单 击 “ 菜 单 ” 进 入 “菜单 ” 
惹 浮 窗 ， 悬 浮 窗 中 列 出 了 当前 系统 所 有 的 菜单 。 找 到 “ 主 菜单 ”， 单 击 其 右 侧 的 “ 列 出 链 
接 ” 链 接 进 入 该 菜单 的 链接 列表 。 我 们 发 现 当 前 “ 主 菜单 ”里 只 有 一 个 名 为 “主页 ”的 链 
接 。 单 击 列 表 上 方 的 “添加 新 链接 ”， 然 后 将 上 面 的 三 个 分 类 添加 进 主 菜 单 中 。 其 中 , “ 菜 
单 链接 标题 ”为 各 分 类 的 名 称 、“ 路 径 ” 为 各 分 类 的 URL 别名 、 权 重 与 各 分 类 在 “类 别 ” 
词汇 表 中 的 权重 相同 。 添 加 完成 后 ， 回 到 “ 主 菜 单 ” 悬 浮 窗 的 “ 列 出 链接 ”页 签 ， 查 看 已 
经 添加 的 各 菜单 链接 ， 如 图 14-33 所 示 。 


。 前 季 撤 保 庆 下 蕴 红 茶 任 。 重 在 下 卉 时 会 全 用 LURL 则 各。/Heerap 会 昌 faxonomyjrerm/5 全 和 
所 的 本 已 经 介 在 。 


图 14-33 ”成 功 添加 分 类 至 主 菜单 中 


当 检查 没 有 问题 之 后 ， 关 闭 “ 主 菜单 ”悬浮 窗 回 到 Drupal 站 点 首页 。 我 们 发 现 之 前 添 
加 的 各 菜单 链接 都 出 现在 了 主 菜单 中 ， 如 图 14-34 所 示 。 单 击 “ 生 活 点 滴 ” 按 钮 ， 也 可 以 
查看 到 《世界 ， 你 好 ! 》 这 篇 文章 。 


图 14-34 已 添加 的 分 类 显示 在 主 菜单 中 


“96° 
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接 下 来 ， 我 们 可 以 把 上 一 节 里 添加 的 那个 “基本 页 面 ”也 做 为 主 菜单 中 的 一 员 添加 进 
来 。 具体 怎么 做 ， 大 家 自己 想 想 吧 。 完 成 以 后 的 效果 如 图 14-35 所 示 。 


图 14-35 添加 “基本 页 面 ”到 主 菜单 


14.2.5 ”管理 用 户 


用 户 管理 对 于 一 个 专业 的 网 站 来 说 是 十 分 必要 的 。 不 同类 型 的 用 户 都 会 接触 到 同一 个 
系统 不 同 的 方面 。 为 了 便于 管理 、 同 时 也 为 了 网 站 和 服务 器 的 安全 ， 对 不 同类 型 的 用 户 给 
予 合适 的 授权 是 一 个 不 错 的 主意 。 

Drupal 对 用 户 权限 的 管理 是 通过 “角色 ”来 实现 的 。 所 谓 “角色 ”， 我 们 可 以 把 它 看 
做 是 拥有 相同 权限 的 一 群 用 户 的 集合 。 通 过 “角色 ”， 我 们 可 以 对 网 站 实现 分 级 管理 ， 同 
时 也 可 以 根据 访问 者 属性 的 差异 ， 为 其 提供 合适 其 阅读 的 内 容 。 

单 击 管理 导航 菜单 中 的 “用 户 ”链接 进 入 “用 户 ” 悬 浮 窗 。 在 这 个 悬浮 窗 中 一 共有 两 
个 页 签 ， 一 个 为 “列表 ”， 另 一 个 为 “权限 ”。 在 “列表 ”页 签 中 ， 我 们 可 以 查看 当前 系 
统 中 所 有 的 用 户 ， 并 可 根据 不 同 的 条 件 过滤 出 需要 查看 的 用 户 ， 如 图 14-36 所 示 。 


图 14-36 Drapal 的 用 户 管理 一 一 列表 


“Ts 
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在 “权限 ”页 签 中 ， 我 们 可 以 为 不 同 角色 的 用 户 分 配 不 同 的 权限 。 同 时 也 可 以 添加 新 
的 角色 ， 如 图 14-37 所 示 。 


让 页 * 全 孚 > 用 P > 和 限 请 


条 四 拓 汪 曙 。 一 个 和 名 一 二 在 5 不 下 上 流 避 有 峙 完 容 的 用 户 。 注 如 第 本 有 有 - 到 名 同 户 ， 认证 用户 所 芝 
上 交 各 和 有 二 闻 。 持 村 角色 尘 和 为 系 少 可 了 《至 分 司 记 ) 应 生生 可 限 ( 竺 诺 肯 ) - 于 和 了 一 个 亲近 沁 生 和 


图 14-37 Drupal 的 用 户 管理 一 一 权限 与 角色 


另外 ， 我 们 还 可 以 通过 管理 导航 菜单 上 的 “配置 ”项 下 的 “用 户 ” 框 中 的 设置 来 决定 
谁 可 以 在 本 站 注册 ， 在 注册 时 是 否 需 要 提供 邮箱 地 址 等 等 与 用 户 注册 有 关 的 内 容 。 我 们 还 
可 以 设置 一 个 IP 地 址 的 黑 名 单 ， 由 这 个 黑 名 单 中 列 出 的 人 P 地 址 发 起 的 访问 都 会 被 拒绝 ， 
从 而 保证 服务 器 的 安全 。 可 以 说 Drupal 的 用 户 管理 功能 灵活 性 很 大 , 但 是 如 果 了 解 了 它 的 
工作 原理 ， 使 用 起 来 还 是 十 分 方便 的 。 


14.2.6 小结 


在 这 一 小 节 里 ， 我 们 了 解 了 Drupal 做 为 一 个 内 容 管 理 框 架 的 各 项 基本 功能 和 使 用 方 
法 。 你 也 可 以 使 用 本 书 中 学 到 的 PHP 知识 ， 结 合 Drupal 网 站 上 的 学 习 资 料 ， 使 用 Drupal 
架设 一 个 功能 完善 、 灵 活性 高 和 安全 性 强 的 网 站 。 

Drupal 和 WordPress 一 样 ， 也 有 着 丰富 的 扩展 和 主题 ， 如 果 你 想 在 Drupal 的 基础 上 进 
行 二 次 开发 也 会 非常 容易 。 


“398。 


